/* 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. */ #undef BACKEND_NAME #define BACKEND_NAME genesys_gl841 #include "genesys_gl841.h" /**************************************************************************** Low level function ****************************************************************************/ /* ------------------------------------------------------------------------ */ /* Read and write RAM, registers and AFE */ /* ------------------------------------------------------------------------ */ /* Write to many registers */ /* Note: There is no known bulk register write, this function is sending single registers instead */ static SANE_Status gl841_bulk_write_register (Genesys_Device * dev, Genesys_Register_Set * reg, size_t elems) { SANE_Status status = SANE_STATUS_GOOD; unsigned int i, c; uint8_t buffer[GENESYS_MAX_REGS * 2]; /* handle differently sized register sets, reg[0x00] is the last one */ i = 0; while ((i < elems) && (reg[i].address != 0)) i++; elems = i; DBG (DBG_io, "gl841_bulk_write_register (elems = %lu)\n", (u_long) elems); for (i = 0; i < elems; i++) { buffer[i * 2 + 0] = reg[i].address; buffer[i * 2 + 1] = reg[i].value; DBG (DBG_io2, "reg[0x%02x] = 0x%02x\n", buffer[i * 2 + 0], buffer[i * 2 + 1]); } for (i = 0; i < elems;) { c = elems - i; if (c > 32) /*32 is max. checked that.*/ c = 32; status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_BUFFER, VALUE_SET_REGISTER, INDEX, c * 2, buffer + i * 2); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_write_register: failed while writing command: %s\n", sane_strstatus (status)); return status; } i += c; } DBG (DBG_io, "gl841_bulk_write_register: wrote %lu registers\n", (u_long) elems); return status; } /* Write bulk data (e.g. shading, gamma) */ static SANE_Status gl841_bulk_write_data (Genesys_Device * dev, uint8_t addr, uint8_t * data, size_t len) { SANE_Status status; size_t size; uint8_t outdata[8]; DBG (DBG_io, "gl841_bulk_write_data writing %lu bytes\n", (u_long) len); status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_REGISTER, VALUE_SET_REGISTER, INDEX, 1, &addr); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_write_data failed while setting register: %s\n", sane_strstatus (status)); return status; } while (len) { if (len > BULKOUT_MAXSIZE) size = BULKOUT_MAXSIZE; else size = len; outdata[0] = BULK_OUT; outdata[1] = BULK_RAM; outdata[2] = VALUE_BUFFER & 0xff; outdata[3] = (VALUE_BUFFER >> 8) & 0xff; outdata[4] = (size & 0xff); outdata[5] = ((size >> 8) & 0xff); outdata[6] = ((size >> 16) & 0xff); outdata[7] = ((size >> 24) & 0xff); status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_BUFFER, VALUE_BUFFER, INDEX, sizeof (outdata), outdata); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_write_data failed while writing command: %s\n", sane_strstatus (status)); return status; } status = sanei_usb_write_bulk (dev->dn, data, &size); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_write_data failed while writing bulk data: %s\n", sane_strstatus (status)); return status; } DBG (DBG_io2, "gl841_bulk_write_data wrote %lu bytes, %lu remaining\n", (u_long) size, (u_long) (len - size)); len -= size; data += size; } DBG (DBG_io, "gl841_bulk_write_data: completed\n"); return status; } /* for debugging transfer rate*/ /* #include <sys/time.h> static struct timeval start_time; static void starttime(){ gettimeofday(&start_time,NULL); } static void printtime(char *p) { struct timeval t; long long int dif; gettimeofday(&t,NULL); dif = t.tv_sec - start_time.tv_sec; dif = dif*1000000 + t.tv_usec - start_time.tv_usec; fprintf(stderr,"%s %lluµs\n",p,dif); } */ /* Read bulk data (e.g. scanned data) */ static SANE_Status gl841_bulk_read_data (Genesys_Device * dev, uint8_t addr, uint8_t * data, size_t len) { SANE_Status status; size_t size, target; uint8_t outdata[8], *buffer; DBG (DBG_io, "gl841_bulk_read_data: requesting %lu bytes\n", (u_long) len); if (len == 0) return SANE_STATUS_GOOD; status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_REGISTER, VALUE_SET_REGISTER, INDEX, 1, &addr); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_read_data failed while setting register: %s\n", sane_strstatus (status)); return status; } outdata[0] = BULK_IN; outdata[1] = BULK_RAM; outdata[2] = VALUE_BUFFER & 0xff; outdata[3] = (VALUE_BUFFER >> 8) & 0xff; outdata[4] = (len & 0xff); outdata[5] = ((len >> 8) & 0xff); outdata[6] = ((len >> 16) & 0xff); outdata[7] = ((len >> 24) & 0xff); status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_BUFFER, VALUE_BUFFER, INDEX, sizeof (outdata), outdata); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_read_data failed while writing command: %s\n", sane_strstatus (status)); return status; } target = len; buffer = data; while (target) { if (target > BULKIN_MAXSIZE) size = BULKIN_MAXSIZE; else size = target; DBG (DBG_io2, "gl841_bulk_read_data: trying to read %lu bytes of data\n", (u_long) size); status = sanei_usb_read_bulk (dev->dn, data, &size); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_bulk_read_data failed while reading bulk data: %s\n", sane_strstatus (status)); return status; } DBG (DBG_io2, "gl841_bulk_read_data read %lu bytes, %lu remaining\n", (u_long) size, (u_long) (target - size)); target -= size; data += size; } if (DBG_LEVEL >= DBG_data && dev->binary!=NULL) { fwrite(buffer, len, 1, dev->binary); } DBGCOMPLETED; return SANE_STATUS_GOOD; } /* Set address for writing data */ static SANE_Status gl841_set_buffer_address_gamma (Genesys_Device * dev, uint32_t addr) { SANE_Status status; DBG (DBG_io, "gl841_set_buffer_address_gamma: setting address to 0x%05x\n", addr & 0xfffffff0); addr = addr >> 4; status = sanei_genesys_write_register (dev, 0x5c, (addr & 0xff)); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_buffer_address_gamma: failed while writing low byte: %s\n", 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, "gl841_set_buffer_address_gamma: failed while writing high byte: %s\n", sane_strstatus (status)); return status; } DBG (DBG_io, "gl841_set_buffer_address_gamma: completed\n"); return status; } /* Write bulk data (e.g. gamma) */ GENESYS_STATIC SANE_Status gl841_bulk_write_data_gamma (Genesys_Device * dev, uint8_t addr, uint8_t * data, size_t len) { SANE_Status status; size_t size; uint8_t outdata[8]; DBG (DBG_io, "gl841_bulk_write_data_gamma writing %lu bytes\n", (u_long) len); status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_REGISTER, VALUE_SET_REGISTER, INDEX, 1, &addr); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "genesys_bulk_write_data_gamma failed while setting register: %s\n", sane_strstatus (status)); return status; } while (len) { if (len > BULKOUT_MAXSIZE) size = BULKOUT_MAXSIZE; else size = len; outdata[0] = BULK_OUT; outdata[1] = BULK_RAM; outdata[2] = 0x00;/* 0x82 works, too */ outdata[3] = 0x00; outdata[4] = (size & 0xff); outdata[5] = ((size >> 8) & 0xff); outdata[6] = ((size >> 16) & 0xff); outdata[7] = ((size >> 24) & 0xff); status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_BUFFER, VALUE_BUFFER, INDEX, sizeof (outdata), outdata); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "genesys_bulk_write_data_gamma failed while writing command: %s\n", sane_strstatus (status)); return status; } status = sanei_usb_write_bulk (dev->dn, data, &size); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "genesys_bulk_write_data_gamma failed while writing bulk data: %s\n", sane_strstatus (status)); return status; } DBG (DBG_io2, "genesys_bulk_write_data:gamma wrote %lu bytes, %lu remaining\n", (u_long) size, (u_long) (len - size)); len -= size; data += size; } DBG (DBG_io, "genesys_bulk_write_data_gamma: completed\n"); return status; } /**************************************************************************** Mid level functions ****************************************************************************/ static SANE_Bool gl841_get_fast_feed_bit (Genesys_Register_Set * regs) { Genesys_Register_Set *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) { Genesys_Register_Set *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) { Genesys_Register_Set *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) { Genesys_Register_Set *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) { Genesys_Register_Set *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, Genesys_Register_Set * regs, SANE_Bool extended, SANE_Bool half_ccd) { Genesys_Register_Set *r; int i; DBG (DBG_proc, "gl841_setup_sensor\n"); /* that one is tricky at least ....*/ r = sanei_genesys_get_address (regs, 0x70); for (i = 0; i < 4; i++, r++) r->value = dev->sensor.regs_0x08_0x0b[i]; r = sanei_genesys_get_address (regs, 0x16); for (i = 0x06; i < 0x0a; i++, r++) r->value = dev->sensor.regs_0x10_0x1d[i]; r = sanei_genesys_get_address (regs, 0x1a); for (i = 0x0a; i < 0x0e; i++, r++) r->value = dev->sensor.regs_0x10_0x1d[i]; r = sanei_genesys_get_address (regs, 0x52); for (i = 0; i < 9; i++, r++) r->value = dev->sensor.regs_0x52_0x5e[i]; /* 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) { /* 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 { /* swap latch times */ r = sanei_genesys_get_address (regs, 0x18); r->value = 0x30; r = sanei_genesys_get_address (regs, 0x52); for (i = 0; i < 6; i++, r++) r->value = dev->sensor.regs_0x52_0x5e[(i + 3) % 6]; 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 */ 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; uint8_t val; uint8_t *data; uint8_t *verify_data; size_t size, verify_size; unsigned int i; DBG (DBG_proc, "sanei_gl841_asic_test\n"); 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, "sanei_gl841_asic_test: failed to write register: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_write_register (dev, 0x39, 0xad); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "sanei_gl841_asic_test: failed to write register: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_read_register (dev, 0x38, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "sanei_gl841_asic_test: failed to read register: %s\n", sane_strstatus (status)); return status; } if (val != 0xde) /* value of register 0x38 */ { DBG (DBG_error, "sanei_gl841_asic_test: register contains invalid value\n"); return SANE_STATUS_IO_ERROR; } status = sanei_genesys_read_register (dev, 0x39, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "sanei_gl841_asic_test: failed to read register: %s\n", sane_strstatus (status)); return status; } if (val != 0xad) /* value of register 0x39 */ { DBG (DBG_error, "sanei_gl841_asic_test: register contains invalid value\n"); 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. */ data = (uint8_t *) malloc (size); if (!data) { DBG (DBG_error, "sanei_gl841_asic_test: could not allocate memory\n"); return SANE_STATUS_NO_MEM; } verify_data = (uint8_t *) malloc (verify_size); if (!verify_data) { free (data); DBG (DBG_error, "sanei_gl841_asic_test: could not allocate memory\n"); return SANE_STATUS_NO_MEM; } 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, "sanei_gl841_asic_test: failed to set buffer address: %s\n", sane_strstatus (status)); free (data); free (verify_data); return status; } /* status = gl841_bulk_write_data (dev, 0x3c, data, size); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "sanei_gl841_asic_test: failed to bulk write data: %s\n", 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, "sanei_gl841_asic_test: failed to set buffer address: %s\n", sane_strstatus (status)); free (data); free (verify_data); return status; } status = gl841_bulk_read_data (dev, 0x45, (uint8_t *) verify_data, verify_size); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "sanei_gl841_asic_test: failed to bulk read data: %s\n", sane_strstatus (status)); free (data); free (verify_data); return status; } /* todo: why i + 2 ? */ for (i = 0; i < size; i++) { if (verify_data[i] != data[i]) { DBG (DBG_error, "sanei_gl841_asic_test: data verification error\n"); 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]); free (data); free (verify_data); return SANE_STATUS_IO_ERROR; } } free (data); free (verify_data); DBG (DBG_info, "sanei_gl841_asic_test: completed\n"); return SANE_STATUS_GOOD; } /* returns the max register bulk size */ static int gl841_bulk_full_size (void) { return GENESYS_GL841_MAX_REGS; } /* * 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; int index=0; 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); INITREG (0x2c, dev->sensor.optical_res>>8); INITREG (0x2d, dev->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[reg_0x6c].value = dev->gpo.value[0]; dev->reg[reg_0x6d].value = dev->gpo.value[1]; dev->reg[reg_0x6e].value = dev->gpo.enable[0]; dev->reg[reg_0x6f].value = dev->gpo.enable[1]; */ dev->reg[reg_0x6b].value |= REG6B_GPO18; dev->reg[reg_0x6b].value &= ~REG6B_GPO17; sanei_gl841_setup_sensor (dev, 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 nr, addr; DBG (DBG_proc, "gl841_init_registers\n"); nr = 0; memset (dev->reg, 0, GENESYS_MAX_REGS * sizeof (Genesys_Register_Set)); if (strcmp (dev->model->name, "canon-lide-80") == 0) { gl841_init_lide80(dev); return ; } for (addr = 1; addr <= 0x0a; addr++) dev->reg[nr++].address = addr; for (addr = 0x10; addr <= 0x27; addr++) dev->reg[nr++].address = addr; dev->reg[nr++].address = 0x29; for (addr = 0x2c; addr <= 0x39; addr++) dev->reg[nr++].address = addr; for (addr = 0x3d; addr <= 0x3f; addr++) dev->reg[nr++].address = addr; for (addr = 0x52; addr <= 0x5a; addr++) dev->reg[nr++].address = addr; for (addr = 0x5d; addr <= 0x87; addr++) dev->reg[nr++].address = addr; dev->reg[reg_0x01].value = 0x20; /* (enable shading), CCD, color, 1M */ if (dev->model->is_cis == SANE_TRUE) { dev->reg[reg_0x01].value |= REG01_CISSET; } else { dev->reg[reg_0x01].value &= ~REG01_CISSET; } dev->reg[reg_0x02].value = 0x30 /*0x38 */ ; /* auto home, one-table-move, full step */ dev->reg[reg_0x02].value |= REG02_AGOHOME; dev->reg[reg_0x02].value |= REG02_MTRPWR; dev->reg[reg_0x02].value |= REG02_FASTFED; dev->reg[reg_0x03].value = 0x1f /*0x17 */ ; /* lamp on */ dev->reg[reg_0x03].value |= REG03_AVEENB; if (dev->model->ccd_type == CCD_PLUSTEK_3600) /* AD front end */ { dev->reg[reg_0x04].value = (2 << REG04S_AFEMOD) | 0x02; } else /* Wolfson front end */ { dev->reg[reg_0x04].value |= 1 << REG04S_AFEMOD; } dev->reg[reg_0x05].value = 0x00; /* disable gamma, 24 clocks/pixel */ if (dev->sensor.sensor_pixels < 0x1500) dev->reg[reg_0x05].value |= REG05_DPIHW_600; else if (dev->sensor.sensor_pixels < 0x2a80) dev->reg[reg_0x05].value |= REG05_DPIHW_1200; else if (dev->sensor.sensor_pixels < 0x5400) dev->reg[reg_0x05].value |= REG05_DPIHW_2400; else { dev->reg[reg_0x05].value |= REG05_DPIHW_2400; DBG (DBG_warn, "gl841_init_registers: Cannot handle sensor pixel count %d\n", dev->sensor.sensor_pixels); } dev->reg[reg_0x06].value |= REG06_PWRBIT; dev->reg[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[reg_0x06].value |= 0 << REG06S_SCANMOD; dev->reg[reg_0x09].value |= 1 << REG09S_CLKSET; } else { dev->reg[reg_0x06].value |= 0x05 << REG06S_SCANMOD; /* 15 clocks/pixel */ dev->reg[reg_0x09].value = 0; /* 24 MHz CLKSET */ } dev->reg[reg_0x1e].value = 0xf0; /* watch-dog time */ dev->reg[reg_0x17].value |= 1 << REG17S_TGW; dev->reg[reg_0x19].value = 0x50; dev->reg[reg_0x1d].value |= 1 << REG1DS_TGSHLD; dev->reg[reg_0x1e].value |= 1 << REG1ES_WDTIME; /*SCANFED*/ dev->reg[reg_0x1f].value = 0x01; /*BUFSEL*/ dev->reg[reg_0x20].value = 0x20; /*LAMPPWM*/ dev->reg[reg_0x29].value = 0xff; /*BWHI*/ dev->reg[reg_0x2e].value = 0x80; /*BWLOW*/ dev->reg[reg_0x2f].value = 0x80; /*LPERIOD*/ dev->reg[reg_0x38].value = 0x4f; dev->reg[reg_0x39].value = 0xc1; /*VSMPW*/ dev->reg[reg_0x58].value |= 3 << REG58S_VSMPW; /*BSMPW*/ dev->reg[reg_0x59].value |= 3 << REG59S_BSMPW; /*RLCSEL*/ dev->reg[reg_0x5a].value |= REG5A_RLCSEL; /*STOPTIM*/ dev->reg[reg_0x5e].value |= 0x2 << REG5ES_STOPTIM; sanei_gl841_setup_sensor (dev, dev->reg, 0, 0); /* set up GPIO */ dev->reg[reg_0x6c].value = dev->gpo.value[0]; dev->reg[reg_0x6d].value = dev->gpo.value[1]; dev->reg[reg_0x6e].value = dev->gpo.enable[0]; dev->reg[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[reg_0x6b].value |= REG6B_GPO18; dev->reg[reg_0x6b].value &= ~REG6B_GPO17; } if (dev->model->gpo_type == GPO_CANONLIDE80) { dev->reg[reg_0x6b].value |= REG6B_GPO18; dev->reg[reg_0x6b].value &= ~REG6B_GPO17; } if (dev->model->gpo_type == GPO_XP300) { dev->reg[reg_0x6b].value |= REG6B_GPO17; } if (dev->model->gpo_type == GPO_DP685) { /* REG6B_GPO18 lights on green led */ dev->reg[reg_0x6b].value |= REG6B_GPO17|REG6B_GPO18; } DBG (DBG_proc, "gl841_init_registers complete\n"); } /* Send slope table for motor movement slope_table in machine byte order */ GENESYS_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; uint8_t *table; char msg[4000]; /*#ifdef WORDS_BIGENDIAN*/ int i; /*#endif*/ DBG (DBG_proc, "gl841_send_slope_table (table_nr = %d, steps = %d)\n", table_nr, steps); dpihw = dev->reg[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; /*#ifdef WORDS_BIGENDIAN*/ table = (uint8_t*)malloc(steps * 2); for(i = 0; i < steps; i++) { table[i * 2] = slope_table[i] & 0xff; table[i * 2 + 1] = slope_table[i] >> 8; } /*#else table = (uint8_t*)slope_table; #endif*/ 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) { /*#ifdef WORDS_BIGENDIAN*/ free(table); /*#endif*/ DBG (DBG_error, "gl841_send_slope_table: failed to set buffer address: %s\n", sane_strstatus (status)); return status; } status = gl841_bulk_write_data (dev, 0x3c, (uint8_t *) table, steps * 2); if (status != SANE_STATUS_GOOD) { /*#ifdef WORDS_BIGENDIAN*/ free(table); /*#endif*/ DBG (DBG_error, "gl841_send_slope_table: failed to send slope table: %s\n", sane_strstatus (status)); return status; } /*#ifdef WORDS_BIGENDIAN*/ free(table); /*#endif*/ DBG (DBG_proc, "gl841_send_slope_table: completed\n"); 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); /* sets to default values */ sanei_genesys_init_fe (dev); /* write them to analog frontend */ status = sanei_genesys_fe_write_data (dev, 0x00, dev->frontend.reg[0]); 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.reg[1]); 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.reg[2]); 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.reg[0]); 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.offset[0]); 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.gain[0]); 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, "gl841_set_ad_fe(): start\n"); if (set == AFE_INIT) { DBG (DBG_proc, "gl841_set_ad_fe(): setting DAC %u\n", dev->model->dac_type); /* sets to default values */ sanei_genesys_init_fe (dev); /* write them to analog frontend */ status = sanei_genesys_fe_write_data (dev, 0x00, dev->frontend.reg[0]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing reg 0x00 failed: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x01, dev->frontend.reg[1]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing reg 0x01 failed: %s\n", 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, "gl841_set_ad_fe: writing sign[%d] failed: %s\n", 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.reg[0]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing reg 0x00 failed: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x01, dev->frontend.reg[1]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing reg 0x01 failed: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x02 (red gain)*/ status = sanei_genesys_fe_write_data (dev, 0x02, dev->frontend.gain[0]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing fe 0x02 (gain r) fail: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x03 (green gain)*/ status = sanei_genesys_fe_write_data (dev, 0x03, dev->frontend.gain[1]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing fe 0x03 (gain g) fail: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x04 (blue gain)*/ status = sanei_genesys_fe_write_data (dev, 0x04, dev->frontend.gain[2]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: writing fe 0x04 (gain b) fail: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x05 (red offset)*/ status = sanei_genesys_fe_write_data (dev, 0x05, dev->frontend.offset[0]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: write fe 0x05 (offset r) fail: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x06 (green offset)*/ status = sanei_genesys_fe_write_data (dev, 0x06, dev->frontend.offset[1]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: write fe 0x06 (offset g) fail: %s\n", sane_strstatus (status)); return status; } /* Write fe 0x07 (blue offset)*/ status = sanei_genesys_fe_write_data (dev, 0x07, dev->frontend.offset[2]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_ad_fe: write fe 0x07 (offset b) fail: %s\n", sane_strstatus (status)); return status; } } DBG (DBG_proc, "gl841_set_ad_fe(): end\n"); return status; } /* Set values of analog frontend */ static SANE_Status gl841_set_fe (Genesys_Device * dev, uint8_t set) { SANE_Status status; int i; DBG (DBG_proc, "gl841_set_fe (%s)\n", set == AFE_INIT ? "init" : set == AFE_SET ? "set" : set == AFE_POWER_SAVE ? "powersave" : "huh?"); /* Analog Device type frontend */ if ((dev->reg[reg_0x04].value & REG04_FESET) == 0x02) { return gl841_set_ad_fe (dev, set); } if ((dev->reg[reg_0x04].value & REG04_FESET) != 0x00) { DBG (DBG_proc, "gl841_set_fe(): unsupported frontend type %d\n", dev->reg[reg_0x04].value & REG04_FESET); return SANE_STATUS_UNSUPPORTED; } if (set == AFE_INIT) { DBG (DBG_proc, "gl841_set_fe(): setting DAC %u\n", dev->model->dac_type); sanei_genesys_init_fe (dev); /* reset only done on init */ status = sanei_genesys_fe_write_data (dev, 0x04, 0x80); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: reset fe failed: %s\n", sane_strstatus (status)); return status; /* if (dev->model->ccd_type == CCD_HP2300 || dev->model->ccd_type == CCD_HP2400) { val = 0x07; status = sanei_usb_control_msg (dev->dn, REQUEST_TYPE_OUT, REQUEST_REGISTER, GPIO_OUTPUT_ENABLE, INDEX, 1, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe failed resetting frontend: %s\n", sane_strstatus (status)); return status; } }*/ } DBG (DBG_proc, "gl841_set_fe(): frontend reset complete\n"); } if (set == AFE_POWER_SAVE) { status = sanei_genesys_fe_write_data (dev, 0x01, 0x02); if (status != SANE_STATUS_GOOD) DBG (DBG_error, "gl841_set_fe: writing data failed: %s\n", sane_strstatus (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.reg[0]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing reg0 failed: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x02, dev->frontend.reg[2]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing reg2 failed: %s\n", sane_strstatus (status)); return status; } } status = sanei_genesys_fe_write_data (dev, 0x01, dev->frontend.reg[1]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing reg1 failed: %s\n", sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x03, dev->frontend.reg[3]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing reg3 failed: %s\n", 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, "gl841_set_fe: writing reg6 failed: %s\n", 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, "gl841_set_fe: writing reg8 failed: %s\n", 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, "gl841_set_fe: writing reg9 failed: %s\n", sane_strstatus (status)); return status; } for (i = 0; i < 3; i++) { status = sanei_genesys_fe_write_data (dev, 0x24 + i, dev->frontend.sign[i]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing sign[%d] failed: %s\n", i, sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x28 + i, dev->frontend.gain[i]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing gain[%d] failed: %s\n", i, sane_strstatus (status)); return status; } status = sanei_genesys_fe_write_data (dev, 0x20 + i, dev->frontend.offset[i]); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_set_fe: writing offset[%d] failed: %s\n", i, sane_strstatus (status)); return status; } } DBG (DBG_proc, "gl841_set_fe: completed\n"); 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; Genesys_Register_Set * r; DBG (DBG_proc, "gl841_init_motor_regs_off : scan_lines=%d\n", 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 */ GENESYS_STATIC SANE_Status gl841_write_freq(Genesys_Device *dev, unsigned int ydpi) { SANE_Status status; /**< 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(gl841_bulk_write_data_gamma (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; } #ifndef UNIT_TESTING static #endif SANE_Status gl841_init_motor_regs(Genesys_Device * dev, 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; 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; Genesys_Register_Set * r; /*number of scan lines to add in a scan_lines line*/ DBG (DBG_proc, "gl841_init_motor_regs : feed_steps=%d, action=%d, flags=%x\n", 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, 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; } #ifndef UNIT_TESTING static #endif SANE_Status gl841_init_motor_regs_scan(Genesys_Device * dev, 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; 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; Genesys_Register_Set * r; unsigned int min_restep = 0x20; uint32_t z1, z2; DBG (DBG_proc, "gl841_init_motor_regs_scan : 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", scan_exposure_time, scan_yres, scan_step_type, scan_lines, scan_dummy, feed_steps, scan_power_mode, flags); fast_exposure = gl841_exposure_time(dev, 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,"gl841_init_motor_regs_scan: Maximum allowed slope steps for fast slope: %d\n",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,"gl841_init_motor_regs_scan: feed too short, slow move forced.\n"); } 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,"gl841_init_motor_regs_scan: Time for slow move: %d\n", slow_time); DBG(DBG_info,"gl841_init_motor_regs_scan: Time for fast move: %d\n", 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,"gl841_init_motor_regs_scan: Decided to use %s mode\n", 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, "gl841_init_motor_regs_scan: z1 = %d\n", z1); DBG (DBG_info, "gl841_init_motor_regs_scan: z2 = %d\n", 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) { Genesys_Register_Set * 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) { Genesys_Register_Set * 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, 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, int color_filter, int flags ) { unsigned int words_per_line; unsigned int end; unsigned int dpiset; unsigned int i; Genesys_Register_Set * r; SANE_Status status; uint16_t expavg, expr, expb, expg; DBG (DBG_proc, "gl841_init_optical_regs_scan : exposure_time=%d, " "used_res=%d, start=%d, pixels=%d, channels=%d, depth=%d, " "half_ccd=%d, flags=%x\n", exposure_time, used_res, start, pixels, channels, depth, half_ccd, flags); end = start + pixels; status = gl841_set_fe (dev, AFE_SET); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_init_optical_regs_scan: failed to set frontend: %s\n", sane_strstatus (status)); return status; } /* adjust used_res for chosen dpihw */ used_res = used_res * gl841_get_dpihw(dev) / 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; if (flags & OPTICAL_FLAG_DISABLE_LAMP) r->value &= ~REG03_LAMPPWR; else r->value |= REG03_LAMPPWR; /* exposure times */ r = sanei_genesys_get_address (reg, 0x10); for (i = 0; i < 6; i++, r++) { if (flags & OPTICAL_FLAG_DISABLE_LAMP) r->value = 0x01;/* 0x0101 is as off as possible */ else { /* EXP[R,G,B] only matter for CIS scanners */ if (dev->sensor.regs_0x10_0x1d[i] == 0x00) r->value = 0x01; /*0x00 will not be accepted*/ else r->value = dev->sensor.regs_0x10_0x1d[i]; } } r = sanei_genesys_get_address (reg, 0x19); if (flags & OPTICAL_FLAG_DISABLE_LAMP) r->value = 0xff; else r->value = 0x50; /* 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 0: r->value |= 0x14; /* red filter */ break; case 1: r->value |= 0x18; /* green filter */ break; case 2: r->value |= 0x1c; /* blue filter */ break; default: r->value |= 0x10; /* no filter */ 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, 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 = dev->sensor.dummy_pixel; DBGCOMPLETED; return SANE_STATUS_GOOD; } static int gl841_get_led_exposure(Genesys_Device * dev) { int d,r,g,b,m; if (!dev->model->is_cis) return 0; d = dev->reg[reg_0x19].value; r = dev->sensor.regs_0x10_0x1d[1] | (dev->sensor.regs_0x10_0x1d[0] << 8); g = dev->sensor.regs_0x10_0x1d[3] | (dev->sensor.regs_0x10_0x1d[2] << 8); b = dev->sensor.regs_0x10_0x1d[5] | (dev->sensor.regs_0x10_0x1d[4] << 8); 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 */ GENESYS_STATIC int gl841_exposure_time(Genesys_Device *dev, 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); 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 */ GENESYS_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 */ GENESYS_STATIC SANE_Status gl841_init_scan_regs (Genesys_Device * dev, Genesys_Register_Set * reg, float xres,/*dpi*/ float yres,/*dpi*/ float startx,/*optical_res, from dummy_pixel+1*/ float starty,/*base_ydpi, from home!*/ float pixels, float lines, unsigned int depth, unsigned int channels, int color_filter, unsigned int flags ) { 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; unsigned int oflags; /**> optical flags */ DBG (DBG_info, "gl841_init_scan_regs settings:\n" "Resolution : %gDPI/%gDPI\n" "Lines : %g\n" "PPL : %g\n" "Startpos : %g/%g\n" "Depth/Channels: %u/%u\n" "Flags : %x\n\n", xres, yres, lines, pixels, startx, starty, depth, channels, flags); /* 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 (dev->sensor.optical_res < 2 * xres || !(dev->model->flags & GENESYS_FLAG_HALF_CCD_MODE)) { half_ccd = SANE_FALSE; } else { half_ccd = SANE_TRUE; } /* optical_res */ optical_res = dev->sensor.optical_res; if (half_ccd) optical_res /= 2; /* stagger */ if ((!half_ccd) && (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE)) stagger = (4 * yres) / dev->motor.base_ydpi; else stagger = 0; DBG (DBG_info, "gl841_init_scan_regs : stagger=%d lines\n", stagger); /* used_res */ i = optical_res / 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 || (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 = ((dev->sensor.CCD_start_xoffset + startx) * used_res) / dev->sensor.optical_res; /* needs to be aligned for used_res */ start = (start * optical_res) / used_res; start += dev->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[reg_0x01].value & REG01_SHDAREA) != 0) { avg=optical_res/used_res; start=(start/avg)*avg; } /* compute correct pixels number */ /* pixels */ used_pixels = (pixels * optical_res) / xres; /* round up pixels number if needed */ if (used_pixels * xres < 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 = yres*channels; else slope_dpi = yres; slope_dpi = slope_dpi * (1 + dummy); scan_step_type = gl841_scan_step_type(dev, yres); exposure_time = gl841_exposure_time(dev, 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(flags & SCAN_FLAG_DYNAMIC_LINEART) { depth=8; } oflags=0; if (flags & SCAN_FLAG_DISABLE_SHADING) { oflags |= OPTICAL_FLAG_DISABLE_SHADING; } if ((flags & SCAN_FLAG_DISABLE_GAMMA) || (depth==16)) { oflags |= OPTICAL_FLAG_DISABLE_GAMMA; } if (flags & SCAN_FLAG_DISABLE_LAMP) { oflags |= OPTICAL_FLAG_DISABLE_LAMP; } if (flags & SCAN_FLAG_ENABLE_LEDADD) { oflags |= OPTICAL_FLAG_ENABLE_LEDADD; } status = gl841_init_optical_regs_scan(dev, reg, exposure_time, used_res, start, used_pixels, channels, depth, half_ccd, 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,channels,yres,flags); /* lincnt */ lincnt = lines + max_shift + stagger; /* add tl_y to base movement */ move = starty; DBG (DBG_info, "gl841_init_scan_regs: move=%d steps\n", move); /* subtract current head position */ move -= dev->scanhead_position_in_steps; DBG (DBG_info, "gl841_init_scan_regs: move=%d steps\n", 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, "gl841_init_scan_regs: move=%d steps\n", move);*/ if (flags & SCAN_FLAG_SINGLE_LINE) status = gl841_init_motor_regs_off(reg, dev->model->is_cis?lincnt*channels:lincnt); else status = gl841_init_motor_regs_scan(dev, reg, exposure_time, slope_dpi, scan_step_type, dev->model->is_cis?lincnt*channels:lincnt, dummy, move, scan_power_mode, (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 * channels * depth) / 8; requested_buffer_size = 8 * bytes_per_line; /* we must use a round number of bytes_per_line */ if (requested_buffer_size > BULKIN_MAXSIZE) requested_buffer_size = (BULKIN_MAXSIZE / bytes_per_line) * bytes_per_line; read_buffer_size = 2 * requested_buffer_size + ((max_shift + stagger) * used_pixels * channels * depth) / 8; RIE(sanei_genesys_buffer_free(&(dev->read_buffer))); RIE(sanei_genesys_buffer_alloc(&(dev->read_buffer), read_buffer_size)); RIE(sanei_genesys_buffer_free(&(dev->lines_buffer))); RIE(sanei_genesys_buffer_alloc(&(dev->lines_buffer), read_buffer_size)); RIE(sanei_genesys_buffer_free(&(dev->shrink_buffer))); RIE(sanei_genesys_buffer_alloc(&(dev->shrink_buffer), requested_buffer_size)); RIE(sanei_genesys_buffer_free(&(dev->out_buffer))); RIE(sanei_genesys_buffer_alloc(&(dev->out_buffer), (8 * dev->settings.pixels * channels * depth) / 8)); dev->read_bytes_left = bytes_per_line * lincnt; DBG (DBG_info, "gl841_init_scan_regs: physical bytes to read = %lu\n", (u_long) dev->read_bytes_left); dev->read_active = SANE_TRUE; dev->current_setup.pixels = (used_pixels * used_res)/optical_res; dev->current_setup.lines = lincnt; dev->current_setup.depth = depth; dev->current_setup.channels = channels; dev->current_setup.exposure_time = exposure_time; dev->current_setup.xres = used_res; dev->current_setup.yres = yres; dev->current_setup.half_ccd = half_ccd; 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 (depth == 1) dev->total_bytes_to_read = ((dev->settings.pixels * dev->settings.lines) / 8 + (((dev->settings.pixels * dev->settings.lines)%8)?1:0) ) * channels; else dev->total_bytes_to_read = dev->settings.pixels * dev->settings.lines * channels * (depth / 8); DBG (DBG_info, "gl841_init_scan_regs: total bytes to send = %lu\n", (u_long) dev->total_bytes_to_read); /* END TODO */ DBG (DBG_proc, "gl841_init_scan_regs: completed\n"); return SANE_STATUS_GOOD; } static SANE_Status gl841_calculate_current_setup (Genesys_Device * dev) { int channels; int depth; int start; float xres;/*dpi*/ float yres;/*dpi*/ float startx;/*optical_res, from dummy_pixel+1*/ float pixels; float lines; 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, "gl841_calculate_current_setup settings:\n" "Resolution: %uDPI\n" "Lines : %u\n" "PPL : %u\n" "Startpos : %.3f/%.3f\n" "Scan mode : %d\n\n", dev->settings.yres, dev->settings.lines, dev->settings.pixels, dev->settings.tl_x, dev->settings.tl_y, dev->settings.scan_mode); /* channels */ if (dev->settings.scan_mode == 4) /* single pass color */ channels = 3; else channels = 1; /* depth */ depth = dev->settings.depth; if (dev->settings.scan_mode == 0) depth = 1; /* start */ start = SANE_UNFIX (dev->model->x_offset); start += dev->settings.tl_x; start = (start * dev->sensor.optical_res) / MM_PER_INCH; xres = dev->settings.xres;/*dpi*/ yres = dev->settings.yres;/*dpi*/ startx = start;/*optical_res, from dummy_pixel+1*/ pixels = dev->settings.pixels; lines = dev->settings.lines; DBG (DBG_info, "gl841_calculate_current_setup settings:\n" "Resolution : %gDPI/%gDPI\n" "Lines : %g\n" "PPL : %g\n" "Startpos : %g\n" "Depth/Channels: %u/%u\n\n", xres, yres, lines, pixels, startx, depth, channels); /* half_ccd */ /* we have 2 domains for ccd: xres below or above half ccd max dpi */ if ((dev->sensor.optical_res < 2 * xres) || !(dev->model->flags & GENESYS_FLAG_HALF_CCD_MODE)) { half_ccd = SANE_FALSE; } else { half_ccd = SANE_TRUE; } /* optical_res */ optical_res = dev->sensor.optical_res; if (half_ccd) optical_res /= 2; /* stagger */ if ((!half_ccd) && (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE)) stagger = (4 * yres) / dev->motor.base_ydpi; else stagger = 0; DBG (DBG_info, "gl841_calculate_current_setup: stagger=%d lines\n", stagger); /* used_res */ i = optical_res / 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 */ /* add x coordinates */ start = ((dev->sensor.CCD_start_xoffset + startx) * used_res) / dev->sensor.optical_res; /* needs to be aligned for used_res */ start = (start * optical_res) / used_res; start += dev->sensor.dummy_pixel + 1; if (stagger > 0) start |= 1; /* compute correct pixels number */ /* pixels */ used_pixels = (pixels * optical_res) / xres; /* round up pixels number if needed */ if (used_pixels * xres < 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 = yres*channels; else slope_dpi = yres; slope_dpi = slope_dpi * (1 + dummy); scan_step_type = gl841_scan_step_type(dev, yres); exposure_time = gl841_exposure_time(dev, 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,channels,yres,0); /* lincnt */ lincnt = lines + max_shift + stagger; dev->current_setup.pixels = (used_pixels * used_res)/optical_res; dev->current_setup.lines = lincnt; dev->current_setup.depth = depth; dev->current_setup.channels = channels; dev->current_setup.exposure_time = exposure_time; dev->current_setup.xres = used_res; dev->current_setup.yres = yres; dev->current_setup.half_ccd = half_ccd; dev->current_setup.stagger = stagger; dev->current_setup.max_shift = max_shift + stagger; DBGCOMPLETED; return SANE_STATUS_GOOD; } static void gl841_set_motor_power (Genesys_Register_Set * regs, SANE_Bool set) { DBG (DBG_proc, "gl841_set_motor_power\n"); if (set) { sanei_genesys_set_reg_from_set (regs, 0x02, sanei_genesys_read_reg_from_set (regs, 0x02) | REG02_MTRPWR); } else { sanei_genesys_set_reg_from_set (regs, 0x02, sanei_genesys_read_reg_from_set (regs, 0x02) & ~REG02_MTRPWR); } } static void gl841_set_lamp_power (Genesys_Device * dev, Genesys_Register_Set * regs, SANE_Bool set) { Genesys_Register_Set * r; int i; if (set) { sanei_genesys_set_reg_from_set (regs, 0x03, sanei_genesys_read_reg_from_set (regs, 0x03) | REG03_LAMPPWR); r = sanei_genesys_get_address (regs, 0x10); for (i = 0; i < 6; i++, r++) { if (dev->sensor.regs_0x10_0x1d[i] == 0x00) r->value = 0x01;/*0x00 will not be accepted*/ else r->value = dev->sensor.regs_0x10_0x1d[i]; } r = sanei_genesys_get_address (regs, 0x19); r->value = 0x50; } else { sanei_genesys_set_reg_from_set (regs, 0x03, sanei_genesys_read_reg_from_set (regs, 0x03) & ~REG03_LAMPPWR); r = sanei_genesys_get_address (regs, 0x10); for (i = 0; i < 6; i++, r++) { r->value = 0x01;/* 0x0101 is as off as possible */ } r = sanei_genesys_get_address (regs, 0x19); r->value = 0xff; } } /*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; DBG(DBG_proc, "gl841_save_power: enable = %d\n", 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); usleep(1000); /*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); usleep(1000); 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[reg_0x6b].value &= ~REG6B_GPO17; dev->calib_reg[reg_0x6b].value &= ~REG6B_GPO17; } gl841_set_fe (dev, 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); usleep(10000); /*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[reg_0x6b].value |= REG6B_GPO17; dev->calib_reg[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[reg_0x6b].value |= REG6B_GPO18; dev->calib_reg[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[reg_0x6b].value |= REG6B_GPO17; dev->calib_reg[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; Genesys_Register_Set local_reg[7]; int rate, exposure_time, tgtime, time; DBG (DBG_proc, "gl841_set_powersaving (delay = %d)\n", delay); local_reg[0].address = 0x01; local_reg[0].value = sanei_genesys_read_reg_from_set (dev->reg, 0x01); /* disable fastmode */ local_reg[1].address = 0x03; local_reg[1].value = sanei_genesys_read_reg_from_set (dev->reg, 0x03); /* Lamp power control */ local_reg[2].address = 0x05; local_reg[2].value = sanei_genesys_read_reg_from_set (dev->reg, 0x05) /*& ~REG05_BASESEL*/; /* 24 clocks/pixel */ local_reg[3].address = 0x18; /* Set CCD type */ local_reg[3].value = 0x00; local_reg[4].address = 0x38; /* line period low */ local_reg[4].value = 0x00; local_reg[5].address = 0x39; /* line period high */ local_reg[5].value = 0x00; local_reg[6].address = 0x1c; /* period times for LPeriod, expR,expG,expB, Z1MODE, Z2MODE */ local_reg[6].value = sanei_genesys_read_reg_from_set (dev->reg, 0x05) & ~REG1C_TGTIME; if (!delay) local_reg[1].value = local_reg[1].value & 0xf0; /* disable lampdog and set lamptime = 0 */ else if (delay < 20) local_reg[1].value = (local_reg[1].value & 0xf0) | 0x09; /* enable lampdog and set lamptime = 1 */ else local_reg[1].value = (local_reg[1].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[1].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[6].value |= tgtime; exposure_time /= rate; if (exposure_time > 65535) exposure_time = 65535; local_reg[4].value = exposure_time >> 8; /* highbyte */ local_reg[5].value = exposure_time & 255; /* lowbyte */ status = gl841_bulk_write_register (dev, local_reg, sizeof (local_reg)/sizeof (local_reg[0])); if (status != SANE_STATUS_GOOD) DBG (DBG_error, "gl841_set_powersaving: failed to bulk write registers: %s\n", sane_strstatus (status)); DBG (DBG_proc, "gl841_set_powersaving: completed\n"); return status; } #ifndef UNIT_TESTING static #endif SANE_Status gl841_start_action (Genesys_Device * dev) { return sanei_genesys_write_register (dev, 0x0f, 0x01); } #ifndef UNIT_TESTING static #endif SANE_Status gl841_stop_action (Genesys_Device * dev) { Genesys_Register_Set local_reg[GENESYS_GL841_MAX_REGS+1]; SANE_Status status; 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; } memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS+1) * sizeof (Genesys_Register_Set)); gl841_init_optical_regs_off(local_reg); gl841_init_motor_regs_off(local_reg,0); status = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); 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; } usleep(100*1000); loop--; } DBGCOMPLETED; return SANE_STATUS_IO_ERROR; } static SANE_Status gl841_get_paper_sensor(Genesys_Device * dev, SANE_Bool * paper_loaded) { SANE_Status status; uint8_t val; status = sanei_genesys_read_register(dev, REG6D, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_get_paper_sensor: failed to read gpio: %s\n", 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[GENESYS_GL841_MAX_REGS+1]; SANE_Status status; uint8_t val; SANE_Bool paper_loaded; unsigned int init_steps; float feed_mm; int loop; DBG (DBG_proc, "gl841_eject_document\n"); if (dev->model->is_sheetfed == SANE_FALSE) { DBG (DBG_proc, "gl841_eject_document: there is no \"eject sheet\"-concept for non sheet fed\n"); DBG (DBG_proc, "gl841_eject_document: finished\n"); return SANE_STATUS_GOOD; } memset (local_reg, 0, sizeof (local_reg)); val = 0; status = sanei_genesys_get_status (dev, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_eject_document: failed to read status register: %s\n", sane_strstatus (status)); return status; } status = gl841_stop_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_eject_document: failed to stop motor: %s\n", sane_strstatus (status)); return SANE_STATUS_IO_ERROR; } memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS+1) * sizeof (Genesys_Register_Set)); gl841_init_optical_regs_off(local_reg); gl841_init_motor_regs(dev,local_reg, 65536,MOTOR_ACTION_FEED,0); status = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_eject_document: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } status = gl841_start_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_eject_document: failed to start motor: %s\n", sane_strstatus (status)); gl841_stop_action (dev); /* send original registers */ gl841_bulk_write_register (dev, dev->reg, GENESYS_GL841_MAX_REGS); return status; } RIE(gl841_get_paper_sensor(dev, &paper_loaded)); if (paper_loaded) { DBG (DBG_info, "gl841_eject_document: paper still loaded\n"); /* 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, "gl841_eject_document: reached home position\n"); DBG (DBG_proc, "gl841_eject_document: finished\n"); break; } usleep (100000); /* sleep 100 ms */ --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, "gl841_eject_document: timeout while waiting for scanhead to go home\n"); 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, "gl841_eject_document: failed to read feed steps: %s\n", 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, "gl841_eject_document: failed to read feed steps: %s\n", sane_strstatus (status)); return status; } DBG (DBG_info, "gl841_eject_document: init_steps: %d, steps: %d\n", init_steps, steps); if (steps > init_steps + (feed_mm * dev->motor.base_ydpi) / MM_PER_INCH) { break; } usleep (100000); /* sleep 100 ms */ ++loop; } status = gl841_stop_action(dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_eject_document: failed to stop motor: %s\n", sane_strstatus (status)); return status; } dev->document = SANE_FALSE; DBG (DBG_proc, "gl841_eject_document: finished\n"); return SANE_STATUS_GOOD; } static SANE_Status gl841_load_document (Genesys_Device * dev) { SANE_Status status; SANE_Bool paper_loaded; int loop = 300; DBG (DBG_proc, "gl841_load_document\n"); while (loop > 0) /* do not wait longer then 30 seconds */ { RIE(gl841_get_paper_sensor(dev, &paper_loaded)); if (paper_loaded) { DBG (DBG_info, "gl841_load_document: document inserted\n"); /* when loading OK, document is here */ dev->document = SANE_TRUE; usleep (1000000); /* give user 1000ms to place document correctly */ break; } usleep (100000); /* sleep 100 ms */ --loop; } if (loop == 0) { /* when we come here then the user needed to much time for this */ DBG (DBG_error, "gl841_load_document: timeout while waiting for document\n"); return SANE_STATUS_IO_ERROR; } DBG (DBG_proc, "gl841_load_document: finished\n"); 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 */ status=sanei_genesys_read_scancnt(dev,&scancnt); 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 == SCAN_MODE_COLOR && 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 ? */ #ifndef UNIT_TESTING static #endif SANE_Status gl841_begin_scan (Genesys_Device * dev, Genesys_Register_Set * reg, SANE_Bool start_motor) { SANE_Status status; Genesys_Register_Set local_reg[4]; uint8_t val; DBG (DBG_proc, "gl841_begin_scan\n"); 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)); } local_reg[0].address = 0x03; if (dev->model->ccd_type != CCD_PLUSTEK_3600) { local_reg[0].value = sanei_genesys_read_reg_from_set (reg, 0x03) | REG03_LAMPPWR; } else { local_reg[0].value = sanei_genesys_read_reg_from_set (reg, 0x03); /* TODO PLUSTEK_3600: why ?? */ } local_reg[1].address = 0x01; local_reg[1].value = sanei_genesys_read_reg_from_set (reg, 0x01) | REG01_SCAN; /* set scan bit */ local_reg[2].address = 0x0d; local_reg[2].value = 0x01; local_reg[3].address = 0x0f; if (start_motor) local_reg[3].value = 0x01; else local_reg[3].value = 0x00; /* do not start motor yet */ status = gl841_bulk_write_register (dev, local_reg, sizeof (local_reg)/sizeof (local_reg[0])); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_begin_scan: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } DBG (DBG_proc, "gl841_begin_scan: completed\n"); return status; } /* Send the stop scan command */ #ifndef UNIT_TESTING static #endif SANE_Status gl841_end_scan (Genesys_Device * dev, Genesys_Register_Set __sane_unused__ * reg, SANE_Bool check_stop) { SANE_Status status; DBG (DBG_proc, "gl841_end_scan (check_stop = %d)\n", 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, "gl841_end_scan: failed to stop: %s\n", sane_strstatus (status)); return status; } } DBG (DBG_proc, "gl841_end_scan: completed\n"); return status; } /* Moves the slider to steps */ #ifndef UNIT_TESTING static #endif SANE_Status gl841_feed (Genesys_Device * dev, int steps) { Genesys_Register_Set local_reg[GENESYS_GL841_MAX_REGS+1]; SANE_Status status; uint8_t val; int loop; DBG (DBG_proc, "gl841_feed (steps = %d)\n", steps); status = gl841_stop_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_feed: failed to stop action: %s\n", sane_strstatus (status)); return status; } memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS+1) * sizeof (Genesys_Register_Set)); gl841_init_optical_regs_off(local_reg); gl841_init_motor_regs(dev,local_reg, steps,MOTOR_ACTION_FEED,0); status = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_feed: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } status = gl841_start_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_feed: failed to start motor: %s\n", sane_strstatus (status)); gl841_stop_action (dev); /* send original registers */ gl841_bulk_write_register (dev, dev->reg, GENESYS_GL841_MAX_REGS); 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, "gl841_feed: failed to read home sensor: %s\n", sane_strstatus (status)); return status; } if (!(val & REG41_MOTORENB)) /* motor enabled */ { DBG (DBG_proc, "gl841_feed: finished\n"); dev->scanhead_position_in_steps += steps; return SANE_STATUS_GOOD; } usleep (100000); /* sleep 100 ms */ ++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, "gl841_feed: timeout while waiting for scanhead to go home\n"); return SANE_STATUS_IO_ERROR; } /* Moves the slider to the home (top) position slowly */ #ifndef UNIT_TESTING static #endif SANE_Status gl841_slow_back_home (Genesys_Device * dev, SANE_Bool wait_until_home) { Genesys_Register_Set local_reg[GENESYS_GL841_MAX_REGS+1]; SANE_Status status; Genesys_Register_Set *r; uint8_t val; int loop = 0; DBG (DBG_proc, "gl841_slow_back_home (wait_until_home = %d)\n", wait_until_home); if (dev->model->is_sheetfed == SANE_TRUE) { DBG (DBG_proc, "gl841_slow_back_home: there is no \"home\"-concept for sheet fed\n"); DBG (DBG_proc, "gl841_slow_back_home: finished\n"); 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, "gl847_slow_back_home: failed to read home sensor: %s\n", sane_strstatus (status)); return status; } if (DBG_LEVEL >= DBG_io) { sanei_genesys_print_status (val); } usleep (100000); /* sleep 100 ms */ /* second is reliable */ status = sanei_genesys_get_status (dev, &val); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_slow_back_home: failed to read home sensor: %s\n", 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, "gl841_slow_back_home: already at home, completed\n"); 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, "gl841_slow_back_home: failed to stop motor: %s\n", sane_strstatus (status)); return SANE_STATUS_IO_ERROR; } } memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS+1) * sizeof (Genesys_Register_Set)); gl841_init_motor_regs(dev,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 (gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS)); status = gl841_start_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_slow_back_home: failed to start motor: %s\n", sane_strstatus (status)); gl841_stop_action (dev); /* send original registers */ gl841_bulk_write_register (dev, dev->reg, GENESYS_GL841_MAX_REGS); 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, "gl841_slow_back_home: failed to read home sensor: %s\n", sane_strstatus (status)); return status; } if (val & REG41_HOMESNR) /* home sensor */ { DBG (DBG_info, "gl841_slow_back_home: reached home position\n"); DBG (DBG_proc, "gl841_slow_back_home: finished\n"); return SANE_STATUS_GOOD; } usleep (100000); /* sleep 100 ms */ ++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, "gl841_slow_back_home: timeout while waiting for scanhead to go home\n"); return SANE_STATUS_IO_ERROR; } DBG (DBG_info, "gl841_slow_back_home: scanhead is still moving\n"); DBG (DBG_proc, "gl841_slow_back_home: finished\n"); 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; uint8_t *data; Genesys_Register_Set local_reg[GENESYS_GL841_MAX_REGS+1]; int steps; int pixels = 600; int dpi = 300; DBGSTART; memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS +1) * sizeof (Genesys_Register_Set)); /* sets for a 200 lines * 600 pixels */ /* normal scan with no shading */ status = gl841_init_scan_regs (dev, local_reg, dpi, dpi, 0, 0,/*we should give a small offset here~60 steps*/ 600, dev->model->search_lines, 8, 1, 1,/*green*/ SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_DISABLE_BUFFER_FULL_MOVE); 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 = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); 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; data = malloc (size); if (!data) { DBG (DBG_error, "gl841_search_start_position: failed to allocate memory\n"); return SANE_STATUS_NO_MEM; } status = gl841_begin_scan (dev, local_reg, SANE_TRUE); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to begin scan: %s\n", 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, size); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to read data: %s\n", sane_strstatus (status)); return status; } if (DBG_LEVEL >= DBG_data) sanei_genesys_write_pnm_file ("search_position.pnm", data, 8, 1, pixels, dev->model->search_lines); status = gl841_end_scan (dev, local_reg, SANE_TRUE); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to end scan: %s\n", sane_strstatus (status)); return status; } /* update regs to copy ASIC internal state */ memcpy (dev->reg, local_reg, (GENESYS_GL841_MAX_REGS + 1) * sizeof (Genesys_Register_Set)); /*TODO: find out where sanei_genesys_search_reference_point stores information, and use that correctly*/ status = sanei_genesys_search_reference_point (dev, data, 0, dpi, pixels, dev->model->search_lines); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to set search reference point: %s\n", sane_strstatus (status)); return status; } free (data); 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) { SANE_Status status; uint8_t channels; uint8_t cksel; DBGSTART; cksel = (dev->calib_reg[reg_0x18].value & REG18_CKSEL) + 1; /* clock speed = 1..4 clocks */ /* set line size */ if (dev->settings.scan_mode == SCAN_MODE_COLOR) /* single pass color */ channels = 3; else channels = 1; status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, dev->settings.yres, 0, 0, dev->sensor.optical_res / cksel, /* XXX STEF XXX !!! */ 20, 16, channels, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE ); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_init_register_for_coarse_calibration: failed to setup scan: %s\n", sane_strstatus (status)); return status; } DBG (DBG_info, "gl841_init_register_for_coarse_calibration: optical sensor res: %d dpi, actual res: %d\n", dev->sensor.optical_res / cksel, dev->settings.xres); status = gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_init_register_for_coarse_calibration: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } /* if (DBG_LEVEL >= DBG_info) sanei_gl841_print_registers (dev->calib_reg);*/ DBGCOMPLETED; return SANE_STATUS_GOOD; } /* init registers for shading calibration */ static SANE_Status gl841_init_regs_for_shading (Genesys_Device * dev) { SANE_Status status; SANE_Int ydpi; float starty=0; DBGSTART; DBG (DBG_proc, "%s: lines = %d\n", __func__, (int)(dev->calib_lines)); /* initial calibration reg values */ memcpy (dev->calib_reg, dev->reg, GENESYS_GL841_MAX_REGS * sizeof (Genesys_Register_Set)); 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 */ starty = 140; } dev->calib_channels = 3; dev->calib_lines = dev->model->shading_lines; status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, ydpi, 0, starty, (dev->sensor.sensor_pixels * dev->settings.xres) / dev->sensor.optical_res, dev->calib_lines, 16, dev->calib_channels, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_USE_OPTICAL_RES | /*SCAN_FLAG_DISABLE_BUFFER_FULL_MOVE |*/ SCAN_FLAG_IGNORE_LINE_DISTANCE); 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 = gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_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) { int channels; int flags; int depth; float move; int move_dpi; float start; SANE_Status status; DBG (DBG_info, "gl841_init_regs_for_scan settings:\nResolution: %uDPI\n" "Lines : %u\nPPL : %u\nStartpos : %.3f/%.3f\nScan mode : %d\n\n", dev->settings.yres, dev->settings.lines, dev->settings.pixels, dev->settings.tl_x, dev->settings.tl_y, dev->settings.scan_mode); gl841_slow_back_home(dev,SANE_TRUE); /* channels */ if (dev->settings.scan_mode == SCAN_MODE_COLOR) /* single pass color */ channels = 3; else channels = 1; /* depth */ depth = dev->settings.depth; if (dev->settings.scan_mode == SCAN_MODE_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, "gl841_init_regs_for_scan: move=%f steps\n", move); move += SANE_UNFIX (dev->model->y_offset); DBG (DBG_info, "gl841_init_regs_for_scan: move=%f steps\n", move); move += dev->settings.tl_y; DBG (DBG_info, "gl841_init_regs_for_scan: move=%f steps\n", move); move = (move * move_dpi) / MM_PER_INCH; /* start */ start = SANE_UNFIX (dev->model->x_offset); start += dev->settings.tl_x; start = (start * dev->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 != SCAN_MODE_COLOR) { DBG (DBG_io, "%s: activating LEDADD\n", __func__); flags |= SCAN_FLAG_ENABLE_LEDADD; } /* enable emulated lineart from gray data */ if(dev->settings.scan_mode == SCAN_MODE_LINEART && dev->settings.dynamic_lineart) { flags |= SCAN_FLAG_DYNAMIC_LINEART; } status = gl841_init_scan_regs (dev, dev->reg, dev->settings.xres, dev->settings.yres, start, move, dev->settings.pixels, dev->settings.lines, depth, channels, dev->settings.color_filter, flags); if (status != SANE_STATUS_GOOD) return status; DBG (DBG_proc, "gl841_init_register_for_scan: completed\n"); 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) { int size; SANE_Status status; uint8_t *gamma; DBGSTART; size = 256; /* allocate temporary gamma tables: 16 bits words, 3 channels */ gamma = (uint8_t *) malloc (size * 2 * 3); if (gamma==NULL) { return SANE_STATUS_NO_MEM; } RIE(sanei_genesys_generate_gamma_buffer(dev, 16, 65535, size, gamma)); /* send address */ status = gl841_set_buffer_address_gamma (dev, 0x00000); if (status != SANE_STATUS_GOOD) { free (gamma); DBG (DBG_error, "gl841_send_gamma_table: failed to set buffer address: %s\n", sane_strstatus (status)); return status; } /* send data */ status = gl841_bulk_write_data_gamma (dev, 0x28, (uint8_t *) gamma, size * 2 * 3); if (status != SANE_STATUS_GOOD) { free (gamma); DBG (DBG_error, "gl841_send_gamma_table: failed to send gamma table: %s\n", sane_strstatus (status)); return status; } free (gamma); 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 */ GENESYS_STATIC SANE_Status gl841_led_calibration (Genesys_Device * dev) { int num_pixels; int total_size; uint8_t *line; int i, j; SANE_Status status = SANE_STATUS_GOOD; int val; int channels; int avg[3], avga, avge; int turn; char fn[20]; uint16_t exp[3], target; Genesys_Register_Set *r; 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; status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, dev->settings.yres, 0, 0, (dev->sensor.sensor_pixels*dev->settings.xres) / dev->sensor.optical_res, 1, 16, channels, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_USE_OPTICAL_RES ); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus (status)); return status; } RIE (gl841_bulk_write_register(dev, dev->calib_reg, GENESYS_GL841_MAX_REGS)); num_pixels = dev->current_setup.pixels; total_size = num_pixels * channels * 2 * 1; /* colors * bytes_per_color * scan lines */ line = malloc (total_size); if (!line) return SANE_STATUS_NO_MEM; /* we try to get equal bright leds here: loop: average per color adjust exposure times */ exp[0] = (dev->sensor.regs_0x10_0x1d[0] << 8) | dev->sensor.regs_0x10_0x1d[1]; exp[1] = (dev->sensor.regs_0x10_0x1d[2] << 8) | dev->sensor.regs_0x10_0x1d[3]; exp[2] = (dev->sensor.regs_0x10_0x1d[4] << 8) | dev->sensor.regs_0x10_0x1d[5]; 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=dev->sensor.gain_white_ref*256; do { dev->sensor.regs_0x10_0x1d[0] = (exp[0] >> 8) & 0xff; dev->sensor.regs_0x10_0x1d[1] = exp[0] & 0xff; dev->sensor.regs_0x10_0x1d[2] = (exp[1] >> 8) & 0xff; dev->sensor.regs_0x10_0x1d[3] = exp[1] & 0xff; dev->sensor.regs_0x10_0x1d[4] = (exp[2] >> 8) & 0xff; dev->sensor.regs_0x10_0x1d[5] = exp[2] & 0xff; r = &(dev->calib_reg[reg_0x10]); for (i = 0; i < 6; i++, r++) { r->value = dev->sensor.regs_0x10_0x1d[i]; RIE (sanei_genesys_write_register (dev, 0x10+i, dev->sensor.regs_0x10_0x1d[i])); } RIE (gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS)); DBG (DBG_info, "%s: starting line reading\n", __func__); RIE (gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE)); RIE (sanei_genesys_read_data_from_scanner (dev, line, total_size)); if (DBG_LEVEL >= DBG_data) { snprintf(fn,20,"led_%d.pnm",turn); sanei_genesys_write_pnm_file (fn, line, 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]); /* cleanup before return */ free (line); 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) { SANE_Status status = SANE_STATUS_GOOD; int num_pixels; int total_size; uint8_t *line; int i; int average; int turn; char fn[20]; 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; } status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, dev->settings.yres, 0, 0, (dev->sensor.sensor_pixels*dev->settings.xres) / dev->sensor.optical_res, 1, 8, 3, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_USE_OPTICAL_RES); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_offset_calibration: failed to setup scan: %s\n", sane_strstatus (status)); return status; } num_pixels = dev->current_setup.pixels; total_size = num_pixels * 3 * 2 * 1; line = malloc (total_size); if (line==NULL) { DBGCOMPLETED; return SANE_STATUS_NO_MEM; } dev->frontend.gain[0] = 0x00; dev->frontend.gain[1] = 0x00; dev->frontend.gain[2] = 0x00; /* loop on scan until target offset is reached */ turn=0; target=24; bottom=0; top=255; do { /* set up offset mid range */ dev->frontend.offset[0] = (top+bottom)/2; dev->frontend.offset[1] = (top+bottom)/2; dev->frontend.offset[2] = (top+bottom)/2; /* scan line */ DBG (DBG_info, "%s: starting line reading\n",__func__); gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS); gl841_set_fe(dev, AFE_SET); gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE); sanei_genesys_read_data_from_scanner (dev, line, total_size); gl841_stop_action (dev); if (DBG_LEVEL >= DBG_data) { snprintf(fn,20,"offset_%02d.pnm",turn); sanei_genesys_write_pnm_file (fn, line, 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); dev->frontend.offset[0]=0; dev->frontend.offset[1]=0; dev->frontend.offset[2]=0; free(line); DBG (DBG_info, "%s: offset=(%d,%d,%d)\n", __func__, dev->frontend.offset[0], dev->frontend.offset[1], dev->frontend.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? */ GENESYS_STATIC SANE_Status gl841_offset_calibration (Genesys_Device * dev) { int num_pixels; int total_size; uint8_t *first_line, *second_line; 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; char fn[20]; SANE_Bool acceptable = SANE_FALSE; int mintgt = 0x400; DBG (DBG_proc, "gl841_offset_calibration\n"); /* Analog Device fronted have a different calibration */ if ((dev->reg[reg_0x04].value & REG04_FESET) == 0x02) { return ad_fe_offset_calibration (dev); } /* offset calibration is always done in color mode */ channels = 3; status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, dev->settings.yres, 0, 0, (dev->sensor.sensor_pixels*dev->settings.xres) / dev->sensor.optical_res, 1, 16, channels, dev->settings.color_filter, 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 ); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_offset_calibration: failed to setup scan: %s\n", sane_strstatus (status)); return status; } num_pixels = dev->current_setup.pixels; total_size = num_pixels * channels * 2 * 1; /* colors * bytes_per_color * scan lines */ first_line = malloc (total_size); if (!first_line) return SANE_STATUS_NO_MEM; second_line = malloc (total_size); if (!second_line) { free (first_line); return SANE_STATUS_NO_MEM; } /* 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.gain[0] = 0x00; dev->frontend.gain[1] = 0x00; dev->frontend.gain[2] = 0x00; offh[0] = 0xff; offh[1] = 0xff; offh[2] = 0xff; offl[0] = 0x00; offl[1] = 0x00; offl[2] = 0x00; turn = 0; do { RIEF2 (gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS), first_line, second_line); for (j=0; j < channels; j++) { off[j] = (offh[j]+offl[j])/2; dev->frontend.offset[j] = off[j]; } status = gl841_set_fe(dev, AFE_SET); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_offset_calibration: failed to setup frontend: %s\n", sane_strstatus (status)); return status; } DBG (DBG_info, "gl841_offset_calibration: starting first line reading\n"); RIEF2 (gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE), first_line, second_line); RIEF2 (sanei_genesys_read_data_from_scanner (dev, first_line, total_size), first_line, second_line); if (DBG_LEVEL >= DBG_data) { snprintf(fn,20,"offset1_%02d.pnm",turn); sanei_genesys_write_pnm_file (fn, first_line, 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,"gl841_offset_calibration: black/white pixels: " "%d/%d,%d/%d,%d/%d\n", 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]; } RIEF2 (gl841_stop_action (dev), first_line, second_line); turn++; } while (!acceptable && turn < 100); DBG(DBG_info,"gl841_offset_calibration: acceptable offsets: %d,%d,%d\n", 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.offset[j] = off[j]; } status = gl841_set_fe(dev, AFE_SET); if (status != SANE_STATUS_GOOD) { free (first_line); free (second_line); DBG (DBG_error, "gl841_offset_calibration: failed to setup frontend: %s\n", sane_strstatus (status)); return status; } DBG (DBG_info, "gl841_offset_calibration: starting second line reading\n"); RIEF2 (gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS), first_line, second_line); RIEF2 (gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE), first_line, second_line); RIEF2 (sanei_genesys_read_data_from_scanner (dev, second_line, total_size), first_line, second_line); if (DBG_LEVEL >= DBG_data) { snprintf(fn,20,"offset2_%02d.pnm",turn); sanei_genesys_write_pnm_file (fn, second_line, 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,"gl841_offset_calibration: black/white pixels: " "%d/%d,%d/%d,%d/%d\n", 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]; } RIEF2 (gl841_stop_action (dev), first_line, second_line); turn++; } while (!acceptable && turn < 100); DBG(DBG_info,"gl841_offset_calibration: acceptable offsets: %d,%d,%d\n", 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,"gl841_offset_calibration: first set: %d/%d,%d/%d,%d/%d\n", off1[0],min1[0],off1[1],min1[1],off1[2],min1[2]); DBG(DBG_info,"gl841_offset_calibration: second set: %d/%d,%d/%d,%d/%d\n", 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,"gl841_offset_calibration: difference too small\n"); 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.offset[j] = off[j]; } DBG(DBG_info,"gl841_offset_calibration: final offsets: %d,%d,%d\n", 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.offset[0] = off[0]; dev->frontend.offset[1] = off[0]; dev->frontend.offset[2] = off[0]; } if (channels == 1) { dev->frontend.offset[1] = dev->frontend.offset[0]; dev->frontend.offset[2] = dev->frontend.offset[0]; } /* cleanup before return */ free (first_line); free (second_line); DBG (DBG_proc, "gl841_offset_calibration: completed\n"); 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. */ GENESYS_STATIC SANE_Status gl841_coarse_gain_calibration (Genesys_Device * dev, int dpi) { int num_pixels; int total_size; uint8_t *line; 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; status = gl841_init_scan_regs (dev, dev->calib_reg, dev->settings.xres, dev->settings.yres, 0, 0, (dev->sensor.sensor_pixels*dev->settings.xres) / dev->sensor.optical_res, lines, 16, channels, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_USE_OPTICAL_RES ); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "%s: failed to setup scan: %s\n", __func__, sane_strstatus (status)); return status; } RIE (gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS)); num_pixels = dev->current_setup.pixels; total_size = num_pixels * channels * 2 * lines; /* colors * bytes_per_color * scan lines */ line = malloc (total_size); if (!line) return SANE_STATUS_NO_MEM; RIEF (gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE), line); RIEF (sanei_genesys_read_data_from_scanner (dev, line, total_size), line); if (DBG_LEVEL >= DBG_data) sanei_genesys_write_pnm_file ("coarse.pnm", line, 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]; 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) dev->frontend.gain[j] = 255; else if (283 - 208/gain[j] < 0) dev->frontend.gain[j] = 0; else dev->frontend.gain[j] = 283 - 208/gain[j]; } else if (dev->model->dac_type == DAC_CANONLIDE80) { dev->frontend.gain[j] = gain[j]*12; } DBG (DBG_proc, "%s: channel %d, max=%d, gain = %f, setting:%d\n", __func__, j, max[j], gain[j],dev->frontend.gain[j]); } 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"); #ifdef SANE_STATUS_HW_LOCKED return SANE_STATUS_HW_LOCKED; #else return SANE_STATUS_JAMMED; #endif } } if (dev->model->is_cis) { if (dev->frontend.gain[0] > dev->frontend.gain[1]) dev->frontend.gain[0] = dev->frontend.gain[1]; if (dev->frontend.gain[0] > dev->frontend.gain[2]) dev->frontend.gain[0] = dev->frontend.gain[2]; dev->frontend.gain[2] = dev->frontend.gain[1] = dev->frontend.gain[0]; } if (channels == 1) { dev->frontend.gain[0] = dev->frontend.gain[1]; dev->frontend.gain[2] = dev->frontend.gain[1]; } free (line); DBG (DBG_info, "%s: gain=(%d,%d,%d)\n", __func__, dev->frontend.gain[0], dev->frontend.gain[1], dev->frontend.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, 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, "sanei_gl841_warmup_lamp\n"); memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS + 1) * sizeof (Genesys_Register_Set)); /* okay.. these should be defaults stored somewhere */ dev->frontend.gain[0] = 0x00; dev->frontend.gain[1] = 0x00; dev->frontend.gain[2] = 0x00; dev->frontend.offset[0] = 0x80; dev->frontend.offset[1] = 0x80; dev->frontend.offset[2] = 0x80; status = gl841_init_scan_regs (dev, local_reg, dev->sensor.optical_res, dev->settings.yres, dev->sensor.dummy_pixel, 0, num_pixels, 1, 16, *channels, dev->settings.color_filter, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_USE_OPTICAL_RES ); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_init_regs_for_warmup: failed to setup scan: %s\n", 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 (gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS)); 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 */ #ifndef UNIT_TESTING static #endif SANE_Status sanei_gl841_repark_head (Genesys_Device * dev) { SANE_Status status; DBG (DBG_proc, "sanei_gl841_repark_head\n"); status = gl841_feed(dev,232); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_repark_head: failed to feed: %s\n", 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, "gl841_park_head: completed\n"); return status; } static SANE_Status gl841_is_compatible_calibration (Genesys_Device * dev, Genesys_Calibration_Cache *cache, int for_overwrite) { SANE_Status status; #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 SANE_STATUS_UNSUPPORTED; } status = gl841_calculate_current_setup (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_is_compatible_calibration: failed to calculate current setup: %s\n", sane_strstatus (status)); return status; } DBG (DBG_proc, "gl841_is_compatible_calibration: checking\n"); if (dev->current_setup.half_ccd != cache->used_setup.half_ccd) return SANE_STATUS_UNSUPPORTED; /* 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 SANE_STATUS_UNSUPPORTED; } } #endif DBGCOMPLETED; return SANE_STATUS_GOOD; } /* * 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; uint8_t val; size_t size; uint8_t *line; int i; 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, "gl841_init: already initialized\n"); DBGCOMPLETED; return SANE_STATUS_GOOD; } } dev->dark_average_data = NULL; dev->white_average_data = NULL; dev->settings.color_filter = 0; /* 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 (gl841_bulk_write_register (dev, dev->reg, GENESYS_GL841_MAX_REGS)); /* Test ASIC and RAM */ if (!(dev->model->flags & GENESYS_FLAG_LAZY_INIT)) { RIE (sanei_gl841_asic_test (dev)); } /* Set analog frontend */ RIE (gl841_set_fe (dev, AFE_INIT)); /* Move home */ RIE (gl841_slow_back_home (dev, SANE_TRUE)); /* Init shading data */ RIE (sanei_genesys_init_shading_data (dev, dev->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, "gl841_init: sanei_gl841_repark_head failed: %s\n", sane_strstatus (status)); return status; } } /* initalize sensor gamma tables */ size = 256; for(i=0;i<3;i++) { if (dev->sensor.gamma_table[i] == NULL) { dev->sensor.gamma_table[i] = (uint16_t *) malloc (2 * size); if (dev->sensor.gamma_table[i] == NULL) { DBG (DBG_error, "gl841_init: could not allocate memory for gamma table %d\n",i); return SANE_STATUS_NO_MEM; } sanei_genesys_create_gamma_table (dev->sensor.gamma_table[i], size, 65535, 65535, dev->sensor.gamma[i]); } } /* send gamma tables */ status = gl841_send_gamma_table (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_init: failed to send initial gamma tables: %s\n", sane_strstatus (status)); return status; } /* initial calibration reg values */ memcpy (dev->calib_reg, dev->reg, (GENESYS_GL841_MAX_REGS + 1) * sizeof (Genesys_Register_Set)); status = gl841_init_scan_regs (dev, dev->calib_reg, 300, 300, 0, 0, (16 * 300) / dev->sensor.optical_res, 1, 16, 3, 0, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA | SCAN_FLAG_SINGLE_LINE | SCAN_FLAG_IGNORE_LINE_DISTANCE | SCAN_FLAG_USE_OPTICAL_RES ); RIE (gl841_bulk_write_register (dev, dev->calib_reg, GENESYS_GL841_MAX_REGS)); size = dev->current_setup.pixels * 3 * 2 * 1; /* colors * bytes_per_color * scan lines */ line = malloc (size); if (!line) return SANE_STATUS_NO_MEM; DBG (DBG_info, "gl841_init: starting dummy data reading\n"); RIEF (gl841_begin_scan (dev, dev->calib_reg, SANE_TRUE), line); sanei_usb_set_timeout(1000);/* 1 second*/ /*ignore errors. next read will succeed*/ sanei_genesys_read_data_from_scanner (dev, line, size); free(line); sanei_usb_set_timeout(30 * 1000);/* 30 seconds*/ RIE (gl841_end_scan (dev, dev->calib_reg, SANE_TRUE)); memcpy (dev->calib_reg, dev->reg, (GENESYS_GL841_MAX_REGS + 1) * sizeof (Genesys_Register_Set)); /* 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)); if (s->val[OPT_SCAN_SW].b == s->last_val[OPT_SCAN_SW].b) s->val[OPT_SCAN_SW].b = (val & 0x01) == 0; if (s->val[OPT_FILE_SW].b == s->last_val[OPT_FILE_SW].b) s->val[OPT_FILE_SW].b = (val & 0x02) == 0; if (s->val[OPT_EMAIL_SW].b == s->last_val[OPT_EMAIL_SW].b) s->val[OPT_EMAIL_SW].b = (val & 0x04) == 0; if (s->val[OPT_COPY_SW].b == s->last_val[OPT_COPY_SW].b) s->val[OPT_COPY_SW].b = (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)); if (s->val[OPT_PAGE_LOADED_SW].b == s->last_val[OPT_PAGE_LOADED_SW].b) s->val[OPT_PAGE_LOADED_SW].b = (val & 0x01) == 0; if (s->val[OPT_SCAN_SW].b == s->last_val[OPT_SCAN_SW].b) s->val[OPT_SCAN_SW].b = (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, SANE_Bool forward, SANE_Bool black) { unsigned int pixels, lines, channels; SANE_Status status; Genesys_Register_Set local_reg[GENESYS_GL841_MAX_REGS + 1]; size_t size; uint8_t *data; int steps, depth, dpi; unsigned int pass, count, found, x, y, length; char title[80]; Genesys_Register_Set *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, "gl841_search_strip %s %s\n", 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.gain[0] = 0xff; dev->frontend.gain[1] = 0xff; dev->frontend.gain[2] = 0xff; } gl841_set_fe (dev, AFE_SET); status = gl841_stop_action (dev); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_search_strip: failed to stop: %s\n", 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 = (dev->sensor.sensor_pixels * dpi) / dev->sensor.optical_res; size = pixels * channels * lines * (depth / 8); data = malloc (size); if (!data) { DBG (DBG_error, "gl841_search_strip: failed to allocate memory\n"); return SANE_STATUS_NO_MEM; } /* 20 cm max length for calibration sheet */ length = ((200 * dpi) / MM_PER_INCH)/lines; dev->scanhead_position_in_steps = 0; memcpy (local_reg, dev->reg, (GENESYS_GL841_MAX_REGS + 1) * sizeof (Genesys_Register_Set)); status = gl841_init_scan_regs (dev, local_reg, dpi, dpi, 0, 0, pixels, lines, depth, channels, 0, SCAN_FLAG_DISABLE_SHADING | SCAN_FLAG_DISABLE_GAMMA); if (status != SANE_STATUS_GOOD) { free(data); 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 = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); if (status != SANE_STATUS_GOOD) { free(data); DBG (DBG_error, "gl841_search_strip: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } status = gl841_begin_scan (dev, local_reg, SANE_TRUE); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_strip: failed to begin scan: %s\n", 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, size); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to read data: %s\n", sane_strstatus (status)); return status; } status = gl841_stop_action (dev); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_strip: gl841_stop_action failed\n"); return status; } pass = 0; if (DBG_LEVEL >= DBG_data) { sprintf (title, "search_strip_%s_%s%02u.pnm", black ? "black" : "white", forward ? "fwd" : "bwd", pass); sanei_genesys_write_pnm_file (title, data, depth, channels, pixels, lines); } /* loop until strip is found or maximum pass number done */ found = 0; while (pass < length && !found) { status = gl841_bulk_write_register (dev, local_reg, GENESYS_GL841_MAX_REGS); if (status != SANE_STATUS_GOOD) { DBG (DBG_error, "gl841_search_strip: failed to bulk write registers: %s\n", sane_strstatus (status)); return status; } /* now start scan */ status = gl841_begin_scan (dev, local_reg, SANE_TRUE); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_strip: failed to begin scan: %s\n", 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, size); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_start_position: failed to read data: %s\n", sane_strstatus (status)); return status; } status = gl841_stop_action (dev); if (status != SANE_STATUS_GOOD) { free (data); DBG (DBG_error, "gl841_search_strip: gl841_stop_action failed\n"); return status; } if (DBG_LEVEL >= DBG_data) { sprintf (title, "search_strip_%s_%s%02u.pnm", black ? "black" : "white", forward ? "fwd" : "bwd", pass); sanei_genesys_write_pnm_file (title, 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, "gl841_search_strip: strip found forward during pass %d at line %d\n", pass, y); } else { DBG (DBG_data, "gl841_search_strip: pixels=%d, count=%d (%d%%)\n", 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, "gl841_search_strip: strip found backward during pass %d \n", pass); } else { DBG (DBG_data, "gl841_search_strip: pixels=%d, count=%d (%d%%)\n", pixels, count, (100 * count) / pixels); } } pass++; } free (data); if (found) { status = SANE_STATUS_GOOD; DBG (DBG_info, "gl841_search_strip: %s strip found\n", black ? "black" : "white"); } else { status = SANE_STATUS_UNSUPPORTED; DBG (DBG_info, "gl841_search_strip: %s strip not found\n", black ? "black" : "white"); } DBG (DBG_proc, "gl841_search_strip: completed\n"); return status; } /** * Send shading calibration data. The buffer is considered to always hold values * for all the channels. */ GENESYS_STATIC SANE_Status gl841_send_shading_data (Genesys_Device * dev, uint8_t * data, int size) { SANE_Status status = SANE_STATUS_GOOD; uint32_t length, x, factor, pixels, i; uint32_t half; uint32_t lines, channels; uint16_t dpiset, dpihw, strpixel ,endpixel, beginpixel; uint8_t *buffer,*ptr,*src; DBGSTART; DBG( DBG_io2, "%s: writing %d bytes of shading data\n",__func__,size); /* old method if no SHDAREA */ if((dev->reg[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); half=dev->current_setup.half_ccd+1; factor=dpihw/dpiset; DBG( DBG_io2, "%s: dpihw=%d, dpiset=%d, half_ccd=%d, factor=%d\n",__func__,dpihw,dpiset,half-1,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 = dev->sensor.CCD_start_xoffset / half; beginpixel += dev->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); buffer=(uint8_t *)malloc(pixels); memset(buffer,0,pixels); /* 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; /* 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 */ RIEF(sanei_genesys_set_buffer_address (dev, 0x5400*i), buffer); RIEF(dev->model->cmd_set->bulk_write_data (dev, 0x3c, buffer, pixels), buffer); } free(buffer); DBGCOMPLETED; return status; } /** the gl841 command set */ static Genesys_Command_Set gl841_cmd_set = { "gl841-generic", /* the name of this set */ 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_bulk_full_size, gl841_set_fe, gl841_set_powersaving, gl841_save_power, gl841_set_motor_power, gl841_set_lamp_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, gl841_slow_back_home, NULL, gl841_bulk_write_register, gl841_bulk_write_data, gl841_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, NULL }; SANE_Status sanei_gl841_init_cmd_set (Genesys_Device * dev) { dev->model->cmd_set = &gl841_cmd_set; return SANE_STATUS_GOOD; } /* vim: set sw=2 cino=>2se-1sn-1s{s^-1st0(0u0 smarttab expandtab: */