sane-project-backends/backend/genesys_gl646.c

5770 wiersze
156 KiB
C

/* 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-2009 Stéphane Voltz <stef.dev@free.fr>
Copyright (C) 2005-2009 Pierre Willenbrock <pierre@pirsoft.dnsalias.org>
Copyright (C) 2007 Luke <iceyfor@gmail.com>
This file is part of the SANE package.
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston,
MA 02111-1307, USA.
As a special exception, the authors of SANE give permission for
additional uses of the libraries contained in this release of SANE.
The exception is that, if you link a SANE library with other files
to produce an executable, this does not by itself cause the
resulting executable to be covered by the GNU General Public
License. Your use of that executable is in no way restricted on
account of linking the SANE library code into it.
This exception does not, however, invalidate any other reasons why
the executable file might be covered by the GNU General Public
License.
If you submit changes to SANE to the maintainers to be included in
a subsequent release, you agree by submitting the changes that
those changes may be distributed with this exception intact.
If you write modifications of your own for SANE, it is your choice
whether to permit this exception to apply to your modifications.
If you do not wish that, delete this exception notice.
*/
#include "genesys_gl646.h"
/**
* returns the value hold by a 3 word register
* @param regs register set from which reading the value
* @param regnum number of the register to read
* @return 24 bit value of the register
*/
static uint32_t
gl646_get_triple_reg (Genesys_Register_Set * regs, int regnum)
{
Genesys_Register_Set *r = NULL;
uint32_t ret = 0;
r = sanei_genesys_get_address (regs, regnum);
ret = r->value;
r = sanei_genesys_get_address (regs, regnum + 1);
ret = (ret << 8) + r->value;
r = sanei_genesys_get_address (regs, regnum + 2);
ret = (ret << 8) + r->value;
return ret;
}
/**
* returns the value hold by a 2 word register
* @param regs register set from which reading the value
* @param regnum number of the register to read
* @return 16 bit value of the register
*/
static uint32_t
gl646_get_double_reg (Genesys_Register_Set * regs, int regnum)
{
Genesys_Register_Set *r = NULL;
uint32_t ret = 0;
r = sanei_genesys_get_address (regs, regnum);
ret = r->value;
r = sanei_genesys_get_address (regs, regnum + 1);
ret = (ret << 8) + r->value;
return ret;
}
/* Write to many registers */
static SANE_Status
gl646_bulk_write_register (Genesys_Device * dev,
Genesys_Register_Set * reg, size_t elems)
{
SANE_Status status;
uint8_t outdata[8];
uint8_t buffer[GENESYS_MAX_REGS * 2];
size_t size;
unsigned int i;
/* handle differently sized register sets, reg[0x00] may be the last one */
i = 0;
while ((i < elems) && (reg[i].address != 0))
i++;
elems = i;
size = i * 2;
DBG (DBG_io, "gl646_bulk_write_register (elems= %lu, size = %lu)\n",
(u_long) elems, (u_long) size);
outdata[0] = BULK_OUT;
outdata[1] = BULK_REGISTER;
outdata[2] = 0x00;
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,
"gl646_bulk_write_register: failed while writing command: %s\n",
sane_strstatus (status));
return status;
}
/* copy registers and values in data buffer */
for (i = 0; i < size; i += 2)
{
buffer[i] = reg[i / 2].address;
buffer[i + 1] = reg[i / 2].value;
}
status = sanei_usb_write_bulk (dev->dn, buffer, &size);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_bulk_write_register: failed while writing bulk data: %s\n",
sane_strstatus (status));
return status;
}
if (DBG_LEVEL >= DBG_io2)
{
for (i = 0; i < size; i += 2)
{
DBG (DBG_io2, "reg[0x%02x] = 0x%02x\n", buffer[i], buffer[i + 1]);
}
/* when full size, decode register content */
if (elems > 60)
{
DBG (DBG_io2, "DPISET =%d\n",
gl646_get_double_reg (reg, REG_DPISET));
DBG (DBG_io2, "DUMMY =%d\n",
sanei_genesys_get_address (reg, REG_DUMMY)->value);
DBG (DBG_io2, "STRPIXEL =%d\n",
gl646_get_double_reg (reg, REG_STRPIXEL));
DBG (DBG_io2, "ENDPIXEL =%d\n",
gl646_get_double_reg (reg, REG_ENDPIXEL));
DBG (DBG_io2, "LINCNT =%d\n",
gl646_get_triple_reg (reg, REG_LINCNT));
DBG (DBG_io2, "MAXWD =%d\n",
gl646_get_triple_reg (reg, REG_MAXWD));
DBG (DBG_io2, "LPERIOD =%d\n",
gl646_get_double_reg (reg, REG_LPERIOD));
DBG (DBG_io2, "FEEDL =%d\n",
gl646_get_triple_reg (reg, REG_FEEDL));
}
}
DBG (DBG_io, "gl646_bulk_write_register: wrote %lu bytes, %lu registers\n",
(u_long) size, (u_long) elems);
return status;
}
/* Write bulk data (e.g. shading, gamma) */
static SANE_Status
gl646_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, "gl646_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,
"gl646_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] = 0x00;
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,
"gl646_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,
"gl646_bulk_write_data failed while writing bulk data: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_io2,
"gl646_bulk_write_data wrote %lu bytes, %lu remaining\n",
(u_long) size, (u_long) (len - size));
len -= size;
data += size;
}
DBG (DBG_io, "gl646_bulk_write_data: end\n");
return status;
}
/**
* reads value from gpio endpoint
*/
static SANE_Status
gl646_gpio_read (SANE_Int dn, uint8_t * value)
{
return sanei_usb_control_msg (dn, REQUEST_TYPE_IN,
REQUEST_REGISTER, GPIO_READ, INDEX, 1, value);
}
/**
* writes the given value to gpio endpoint
*/
static SANE_Status
gl646_gpio_write (SANE_Int dn, uint8_t value)
{
DBG (DBG_proc, "gl646_gpio_write(0x%02x)\n", value);
return sanei_usb_control_msg (dn, REQUEST_TYPE_OUT,
REQUEST_REGISTER, GPIO_WRITE,
INDEX, 1, &value);
}
/**
* writes the given value to gpio output enable endpoint
*/
static SANE_Status
gl646_gpio_output_enable (SANE_Int dn, uint8_t value)
{
DBG (DBG_proc, "gl646_gpio_output_enable(0x%02x)\n", value);
return sanei_usb_control_msg (dn, REQUEST_TYPE_OUT,
REQUEST_REGISTER, GPIO_OUTPUT_ENABLE,
INDEX, 1, &value);
}
/* Read bulk data (e.g. scanned data) */
static SANE_Status
gl646_bulk_read_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, "gl646_bulk_read_data: requesting %lu bytes\n", (u_long) len);
/* write requested size */
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,
"gl646_bulk_read_data failed while setting register: %s\n",
sane_strstatus (status));
return status;
}
outdata[0] = BULK_IN;
outdata[1] = BULK_RAM;
outdata[2] = 0x00;
outdata[3] = 0x00;
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,
"gl646_bulk_read_data failed while writing command: %s\n",
sane_strstatus (status));
return status;
}
while (len)
{
if (len > GL646_BULKIN_MAXSIZE)
size = GL646_BULKIN_MAXSIZE;
else
size = len;
DBG (DBG_io2,
"gl646_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,
"gl646_bulk_read_data failed while reading bulk data: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_io2,
"gl646_bulk_read_data read %lu bytes, %lu remaining\n",
(u_long) size, (u_long) (len - size));
len -= size;
data += size;
}
if (dev->model->is_sheetfed == SANE_TRUE)
{
gl646_detect_document_end (dev);
}
DBG (DBG_io, "gl646_bulk_read_data: end\n");
return status;
}
#if 0
static SANE_Status
read_triple_reg (Genesys_Device * dev, int index, unsigned int *words)
{
SANE_Status status;
uint8_t value;
DBG (DBG_proc, "read_triple_reg\n");
RIE (sanei_genesys_read_register (dev, index + 2, &value));
*words = value;
RIE (sanei_genesys_read_register (dev, index + 1, &value));
*words += (value * 256);
RIE (sanei_genesys_read_register (dev, index, &value));
if (dev->model->asic_type == GENESYS_GL646)
*words += ((value & 0x03) * 256 * 256);
else
*words += ((value & 0x0f) * 256 * 256);
DBG (DBG_proc, "read_triple_reg: value=%d\n", *words);
return status;
}
#endif
static SANE_Bool
gl646_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
gl646_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
gl646_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
gl646_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
gl646_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
gl646_test_buffer_empty_bit (SANE_Byte val)
{
if (val & REG41_BUFEMPTY)
return SANE_TRUE;
return SANE_FALSE;
}
static SANE_Bool
gl646_test_motor_flag_bit (SANE_Byte val)
{
if (val & REG41_MOTMFLG)
return SANE_TRUE;
return SANE_FALSE;
}
static void
gl646_set_triple_reg (Genesys_Register_Set * regs, int regnum, uint32_t value)
{
Genesys_Register_Set *r = NULL;
r = sanei_genesys_get_address (regs, regnum);
r->value = LOBYTE (HIWORD (value));
r = sanei_genesys_get_address (regs, regnum + 1);
r->value = HIBYTE (LOWORD (value));
r = sanei_genesys_get_address (regs, regnum + 2);
r->value = LOBYTE (LOWORD (value));
}
static void
gl646_set_double_reg (Genesys_Register_Set * regs, int regnum, uint16_t value)
{
Genesys_Register_Set *r = NULL;
r = sanei_genesys_get_address (regs, regnum);
r->value = HIBYTE (LOWORD (value));
r = sanei_genesys_get_address (regs, regnum + 1);
r->value = LOBYTE (LOWORD (value));
}
/**
* decodes and prints content of status (0x41) register
* @param val value read from reg41
*/
static void
print_status (uint8_t val)
{
char msg[80];
sprintf (msg, "%s%s%s%s%s%s%s%s",
val & REG41_PWRBIT ? "PWRBIT " : "",
val & REG41_BUFEMPTY ? "BUFEMPTY " : "",
val & REG41_FEEDFSH ? "FEEDFSH " : "",
val & REG41_SCANFSH ? "SCANFSH " : "",
val & REG41_HOMESNR ? "HOMESNR " : "",
val & REG41_LAMPSTS ? "LAMPSTS " : "",
val & REG41_FEBUSY ? "FEBUSY " : "",
val & REG41_MOTMFLG ? "MOTMFLG" : "");
DBG (DBG_info, "status=%s\n", msg);
}
/**
* start scanner's motor
* @param dev scanner's device
*/
static SANE_Status
gl646_start_motor (Genesys_Device * dev)
{
return sanei_genesys_write_register (dev, 0x0f, 0x01);
}
/**
* stop scanner's motor
* @param dev scanner's device
*/
static SANE_Status
gl646_stop_motor (Genesys_Device * dev)
{
return sanei_genesys_write_register (dev, 0x0f, 0x00);
}
/**
* find the lowest resolution for the sensor in the given mode.
* @param sensor id of the sensor
* @param required required resolution
* @param color true is color mode
* @return the closest resolution for the sensor and mode
*/
static int
get_lowest_resolution (int sensor, SANE_Bool color)
{
int i, nb;
int dpi;
i = 0;
dpi = 9600;
nb = sizeof (sensor_master) / sizeof (Sensor_Master);
while (sensor_master[i].sensor != -1 && i < nb)
{
/* computes distance and keep mode if it is closer than previous */
if (sensor == sensor_master[i].sensor
&& sensor_master[i].color == color)
{
if (sensor_master[i].dpi < dpi)
{
dpi = sensor_master[i].dpi;
}
}
i++;
}
DBG (DBG_info, "get_lowest_resolution: %d\n", dpi);
return dpi;
}
/**
* find the closest match in mode tables for the given resolution and scan mode.
* @param sensor id of the sensor
* @param required required resolution
* @param color true is color mode
* @return the closest resolution for the sensor and mode
*/
static int
get_closest_resolution (int sensor, int required, SANE_Bool color)
{
int i, nb;
int dist, dpi;
i = 0;
dpi = 0;
dist = 9600;
nb = sizeof (sensor_master) / sizeof (Sensor_Master);
while (sensor_master[i].sensor != -1 && i < nb)
{
/* exit on perfect match */
if (sensor == sensor_master[i].sensor
&& sensor_master[i].dpi == required
&& sensor_master[i].color == color)
{
DBG (DBG_info, "get_closest_resolution: match found for %d\n",
required);
return required;
}
/* computes distance and keep mode if it is closer than previous */
if (sensor == sensor_master[i].sensor
&& sensor_master[i].color == color)
{
if (abs (sensor_master[i].dpi - required) < dist)
{
dpi = sensor_master[i].dpi;
dist = abs (sensor_master[i].dpi - required);
}
}
i++;
}
DBG (DBG_info, "get_closest_resolution: closest match for %d is %d\n",
required, dpi);
return dpi;
}
/**
* Computes if sensor will be set up for half ccd pixels for the given
* scan mode.
* @param sensor id of the sensor
* @param required required resolution
* @param color true is color mode
* @return SANE_TRUE if half ccd is used
*/
static SANE_Bool
is_half_ccd (int sensor, int required, SANE_Bool color)
{
int i, nb;
i = 0;
nb = sizeof (sensor_master) / sizeof (Sensor_Master);
while (sensor_master[i].sensor != -1 && i < nb)
{
/* exit on perfect match */
if (sensor == sensor_master[i].sensor
&& sensor_master[i].dpi == required
&& sensor_master[i].color == color)
{
DBG (DBG_io, "is_half_ccd: match found for %d (half_ccd=%d)\n",
required, sensor_master[i].half_ccd);
return sensor_master[i].half_ccd;
}
i++;
}
DBG (DBG_info, "is_half_ccd: failed to find match for %d dpi\n", required);
return SANE_FALSE;
}
/**
* Returns the cksel values used by the required scan mode.
* @param sensor id of the sensor
* @param required required resolution
* @param color true is color mode
* @return cksel value for mode
*/
static int
get_cksel (int sensor, int required, SANE_Bool color)
{
int i, nb;
i = 0;
nb = sizeof (sensor_master) / sizeof (Sensor_Master);
while (sensor_master[i].sensor != -1 && i < nb)
{
/* exit on perfect match */
if (sensor == sensor_master[i].sensor
&& sensor_master[i].dpi == required
&& sensor_master[i].color == color)
{
DBG (DBG_io, "get_cksel: match found for %d (cksel=%d)\n",
required, sensor_master[i].cksel);
return sensor_master[i].cksel;
}
i++;
}
DBG (DBG_error, "get_cksel: failed to find match for %d dpi\n", required);
/* fail safe fallback */
return 1;
}
/**
* Setup register and motor tables for a scan at the
* given resolution and color mode. TODO try to not use any filed from
* the device.
* @param dev pointer to a struct describing the device
* @param regs register set to fill
* @param scan_settings scan's settings
* @param slope_table1 first motor table to fill
* @param slope_table2 second motor table to fill
* @param resolution dpi of the scan
* @param move distance to move (at scan's dpi) before scan
* @param linecnt number of lines to scan at scan's dpi
* @param startx start of scan area on CCD at CCD's optical resolution
* @param endx end of scan area on CCD at CCD's optical resolution
* @param color SANE_TRUE is color scan
* @param depth 1, 8 or 16 bits data sample
* @return SANE_STATUS_GOOD if registers could be set, SANE_STATUS_INVAL if
* conditions can't be met.
* @note No harcoded SENSOR or MOTOR 'names' should be present and
* registers are set from settings tables and flags related
* to the hardware capabilities.
* */
static SANE_Status
gl646_setup_registers (Genesys_Device * dev,
Genesys_Register_Set * regs,
Genesys_Settings scan_settings,
uint16_t * slope_table1,
uint16_t * slope_table2,
SANE_Int resolution,
uint32_t move,
uint32_t linecnt,
uint16_t startx,
uint16_t endx, SANE_Bool color, SANE_Int depth)
{
SANE_Status status = SANE_STATUS_GOOD;
int i, nb;
Sensor_Master *sensor = NULL;
Motor_Master *motor = NULL;
Sensor_Settings *settings = NULL;
Genesys_Register_Set *r;
unsigned int used1, used2, vfinal;
uint32_t z1, z2;
uint16_t ex, sx;
int channels = 1, stagger, wpl, max_shift;
size_t requested_buffer_size;
size_t read_buffer_size;
SANE_Bool half_ccd = SANE_FALSE;
SANE_Int xresolution;
int feedl;
DBG (DBG_proc, "gl646_setup_registers: start\n");
DBG (DBG_info, "gl646_setup_registers: startx=%d, endx=%d, linecnt=%d\n",
startx, endx, linecnt);
/* x resolution is capped by sensor's capability */
if (resolution > dev->sensor.optical_res)
{
xresolution = dev->sensor.optical_res;
}
else
{
xresolution = resolution;
}
/* for the given resolution, search for master
* sensor mode setting */
i = 0;
nb = sizeof (sensor_master) / sizeof (Sensor_Master);
while (sensor_master[i].sensor != -1 && i < nb)
{
if (dev->model->ccd_type == sensor_master[i].sensor
&& sensor_master[i].dpi == xresolution
&& sensor_master[i].color == color)
{
sensor = &sensor_master[i];
}
i++;
}
if (sensor == NULL)
{
DBG (DBG_error,
"gl646_setup_registers: unable to find settings for sensor %d at %d dpi color=%d\n",
dev->model->ccd_type, xresolution, color);
return SANE_STATUS_INVAL;
}
/* for the given resolution, search for master
* motor mode setting */
i = 0;
nb = sizeof (motor_master) / sizeof (Motor_Master);
while (motor_master[i].motor != -1 && i < nb)
{
if (dev->model->motor_type == motor_master[i].motor
&& motor_master[i].dpi == resolution
&& motor_master[i].color == color)
{
motor = &motor_master[i];
}
i++;
}
if (motor == NULL)
{
DBG (DBG_error,
"gl646_setup_registers: unable to find settings for motor %d at %d dpi, color=%d\n",
dev->model->motor_type, resolution, color);
return SANE_STATUS_INVAL;
}
/* now we can search for the specific sensor settings */
i = 0;
nb = sizeof (sensor_settings) / sizeof (Sensor_Settings);
while (sensor_settings[i].sensor != -1 && i < nb)
{
if (sensor->sensor == sensor_settings[i].sensor
&& sensor->cksel == sensor_settings[i].cksel)
{
settings = &sensor_settings[i];
}
i++;
}
if (settings == NULL)
{
DBG (DBG_error,
"gl646_setup_registers: unable to find settings for sensor %d with '%d' ccd timing\n",
sensor->sensor, sensor->cksel);
return SANE_STATUS_INVAL;
}
/* half_ccd if manual clock programming or dpi is half dpiset */
half_ccd = sensor->half_ccd;
/* now apply values from settings to registers */
if (sensor->regs_0x10_0x15 != NULL)
{
for (i = 0; i < 6; i++)
{
r = sanei_genesys_get_address (regs, 0x10 + i);
r->value = sensor->regs_0x10_0x15[i];
}
}
else
{
for (i = 0; i < 6; i++)
{
r = sanei_genesys_get_address (regs, 0x10 + i);
r->value = 0;
}
}
for (i = 0; i < 4; i++)
{
r = sanei_genesys_get_address (regs, 0x08 + i);
if (half_ccd == SANE_TRUE)
r->value = settings->manual_0x08_0x0b[i];
else
r->value = settings->regs_0x08_0x0b[i];
}
for (i = 0; i < 8; i++)
{
r = sanei_genesys_get_address (regs, 0x16 + i);
r->value = settings->regs_0x16_0x1d[i];
}
for (i = 0; i < 13; i++)
{
r = sanei_genesys_get_address (regs, 0x52 + i);
r->value = settings->regs_0x52_0x5e[i];
}
if (half_ccd == SANE_TRUE)
{
for (i = 0; i < 7; i++)
{
r = sanei_genesys_get_address (regs, 0x52 + i);
r->value = settings->manual_0x52_0x58[i];
}
}
/* now generate slope tables : we are not using generate_slope_table3 yet */
sanei_genesys_generate_slope_table (slope_table1, motor->steps1,
motor->steps1 + 1, motor->vend1,
motor->vstart1, motor->vend1,
motor->steps1, motor->g1, &used1,
&vfinal);
sanei_genesys_generate_slope_table (slope_table2, motor->steps2,
motor->steps2 + 1, motor->vend2,
motor->vstart2, motor->vend2,
motor->steps2, motor->g2, &used2,
&vfinal);
if (color == SANE_TRUE)
channels = 3;
else
channels = 1;
/* R01 */
/* now setup other registers for final scan (ie with shading enabled) */
/* watch dog + shading + scan enable */
regs[reg_0x01].value |= REG01_DOGENB | REG01_DVDSET | REG01_SCAN;
if (dev->model->is_cis == SANE_TRUE)
regs[reg_0x01].value |= REG01_CISSET;
else
regs[reg_0x01].value &= ~REG01_CISSET;
/* if device has no calibration, don't enable shading correction */
if (dev->model->flags & GENESYS_FLAG_NO_CALIBRATION)
{
regs[reg_0x01].value &= ~REG01_DVDSET;
}
regs[reg_0x01].value &= ~REG01_FASTMOD;
if (motor->fastmod)
regs[reg_0x01].value |= REG01_FASTMOD;
/* R02 */
/* allow moving when buffer full by default */
if (dev->model->is_sheetfed == SANE_FALSE)
dev->reg[reg_0x02].value &= ~REG02_ACDCDIS;
else
dev->reg[reg_0x02].value |= REG02_ACDCDIS;
/* setup motor power and direction */
regs[reg_0x02].value |= REG02_MTRPWR;
regs[reg_0x02].value &= ~REG02_MTRREV;
/* fastfed enabled (2 motor slope tables) */
if (motor->fastfed)
regs[reg_0x02].value |= REG02_FASTFED;
else
regs[reg_0x02].value &= ~REG02_FASTFED;
/* step type */
regs[reg_0x02].value &= ~REG02_STEPSEL;
switch (motor->steptype)
{
case FULL_STEP:
break;
case HALF_STEP:
regs[reg_0x02].value |= 1;
break;
case QUATER_STEP:
regs[reg_0x02].value |= 2;
break;
default:
regs[reg_0x02].value |= 3;
break;
}
/* if sheetfed, no AGOHOME */
if (dev->model->is_sheetfed == SANE_TRUE)
{
regs[reg_0x02].value &= ~REG02_AGOHOME;
}
else
{
regs[reg_0x02].value |= REG02_AGOHOME;
}
/* R03 */
regs[reg_0x03].value &= ~REG03_AVEENB;
/* regs[reg_0x03].value |= REG03_AVEENB; */
regs[reg_0x03].value &= ~REG03_LAMPDOG;
/* select XPA */
regs[reg_0x03].value &= ~REG03_XPASEL;
if (scan_settings.scan_method == SCAN_METHOD_TRANSPARENCY)
{
regs[reg_0x03].value |= REG03_XPASEL;
}
/* R04 */
if (depth > 8)
{
regs[reg_0x04].value |= REG04_BITSET;
}
else
{
regs[reg_0x04].value &= ~REG04_BITSET;
}
/* R05 */
regs[reg_0x05].value &= ~REG05_DPIHW;
switch (dev->sensor.optical_res)
{
case 600:
regs[reg_0x05].value |= REG05_DPIHW_600;
break;
case 1200:
regs[reg_0x05].value |= REG05_DPIHW_1200;
break;
case 2400:
regs[reg_0x05].value |= REG05_DPIHW_2400;
break;
default:
regs[reg_0x05].value |= REG05_DPIHW;
}
/* gamma enable for scans */
if (dev->model->flags & GENESYS_FLAG_14BIT_GAMMA)
regs[reg_0x05].value |= REG05_GMM14BIT;
if (depth < 16)
regs[reg_0x05].value |= REG05_GMMENB;
else
regs[reg_0x05].value &= ~REG05_GMMENB;
/* true CIS gray if needed */
if (dev->model->is_cis == SANE_TRUE && color == SANE_FALSE
&& dev->settings.true_gray)
{
regs[reg_0x05].value |= REG05_LEDADD;
}
else
{
regs[reg_0x05].value &= ~REG05_LEDADD;
}
/* cktoggle, ckdelay and cksel at once, cktdelay=2 => half_ccd for md5345 */
regs[reg_0x18].value = sensor->reg_0x18;
/* manual CCD/2 clock programming => half_ccd for hp2300 */
regs[reg_0x1d].value = sensor->reg_0x1d;
/* motor steps used */
regs[reg_0x21].value = motor->steps1;
regs[reg_0x22].value = motor->fwdbwd;
regs[reg_0x23].value = motor->fwdbwd;
regs[reg_0x24].value = motor->steps1;
/* we adjust linecnt according to real motor dpi */
linecnt = (linecnt * motor->ydpi) / scan_settings.yres;
/* scanned area height must be enlarged by max color shift needed */
/* all values are assumed >= 0 */
if (channels > 1)
{
max_shift = dev->model->ld_shift_r;
if (dev->model->ld_shift_b > max_shift)
max_shift = dev->model->ld_shift_b;
if (dev->model->ld_shift_g > max_shift)
max_shift = dev->model->ld_shift_g;
max_shift = (max_shift * scan_settings.yres) / dev->motor.base_ydpi;
linecnt += max_shift;
}
else
{
max_shift = 0;
}
/* at QUATER_STEP lines are 'staggered' and need correction */
stagger = 0;
if ((!half_ccd) && (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE))
{
/* for HP3670, stagger happens only at >=1200 dpi */
if ((dev->model->motor_type != MOTOR_HP3670
&& dev->model->motor_type != MOTOR_HP2400)
|| scan_settings.yres >= dev->sensor.optical_res)
{
stagger = (4 * scan_settings.yres) / dev->motor.base_ydpi;
}
}
linecnt += stagger;
DBG (DBG_info, "gl646_setup_registers : max_shift=%d, stagger=%d lines\n",
max_shift, stagger);
/* CIS scanners read one line per color channel
* since gray mode use 'add' we also read 3 channels even not in
* color mode */
if (dev->model->is_cis == SANE_TRUE)
{
gl646_set_triple_reg (regs, REG_LINCNT, linecnt * 3);
linecnt *= channels;
}
else
{
gl646_set_triple_reg (regs, REG_LINCNT, linecnt);
}
/* scanner's x coordinates are expressed in physical DPI but they must be divided by cksel */
sx = startx / sensor->cksel;
ex = endx / sensor->cksel;
if (half_ccd == SANE_TRUE)
{
sx /= 2;
ex /= 2;
}
gl646_set_double_reg (regs, REG_STRPIXEL, sx);
gl646_set_double_reg (regs, REG_ENDPIXEL, ex);
DBG (DBG_info, "gl646_setup_registers: startx=%d, endx=%d, half_ccd=%d\n",
sx, ex, half_ccd);
/* wpl must be computed according to the scan's resolution */
/* in fact, wpl _gives_ the actual scan resolution */
wpl = (((endx - startx) * sensor->xdpi) / dev->sensor.optical_res);
if (depth == 16)
wpl *= 2;
if (dev->model->is_cis == SANE_FALSE)
wpl *= channels;
dev->wpl = wpl;
DBG (DBG_info, "gl646_setup_registers: wpl=%d\n", wpl);
gl646_set_triple_reg (regs, REG_MAXWD, wpl);
gl646_set_double_reg (regs, REG_DPISET, sensor->dpiset);
gl646_set_double_reg (regs, REG_LPERIOD, sensor->exposure);
/* move distance must be adjusted to take into account the extra lines
* read to reorder data */
feedl = move;
if (stagger + max_shift > 0 && feedl != 0)
{
if (feedl >
((max_shift + stagger) * dev->motor.optical_ydpi) / motor->ydpi)
feedl =
feedl -
((max_shift + stagger) * dev->motor.optical_ydpi) / motor->ydpi;
}
/* we assume all scans are done with 2 tables */
/*
feedl = feed_steps - fast_slope_steps*2 -
(slow_slope_steps >> scan_step_type); */
/* but head has moved due to shading calibration => dev->scanhead_position_in_steps */
if (feedl > 0)
{
/* take into account the distance moved during calibration */
/* feedl -= dev->scanhead_position_in_steps; */
DBG (DBG_info, "gl646_setup_registers: initial move=%d\n", feedl);
DBG (DBG_info, "gl646_setup_registers: scanhead_position_in_steps=%d\n",
dev->scanhead_position_in_steps);
/* TODO clean up this when I'll fully understand.
* for now, special casing each motor */
switch (dev->model->motor_type)
{
case MOTOR_5345:
switch (motor->ydpi)
{
case 200:
feedl -= 70;
break;
case 300:
feedl -= 70;
break;
case 400:
feedl += 130;
break;
case 600:
feedl += 160;
break;
case 1200:
feedl += 160;
break;
case 2400:
feedl += 180;
break;
default:
break;
}
break;
case MOTOR_HP2300:
switch (motor->ydpi)
{
case 75:
feedl -= 180;
break;
case 150:
feedl += 0;
break;
case 300:
feedl += 30;
break;
case 600:
feedl += 35;
break;
case 1200:
feedl += 45;
break;
default:
break;
}
break;
/* theorical value */
default:
if (motor->fastfed)
{
feedl =
feedl - 2 * motor->steps2 -
(motor->steps1 >> motor->steptype);
}
else
{
feedl = feedl - (motor->steps1 >> motor->steptype);
}
break;
}
/* security */
if (feedl < 0)
feedl = 0;
}
DBG (DBG_info, "gl646_setup_registers: final move=%d\n", feedl);
gl646_set_triple_reg (regs, REG_FEEDL, feedl);
regs[reg_0x65].value = motor->mtrpwm;
sanei_genesys_calculate_zmode2 (regs[reg_0x02].value & REG02_FASTFED,
sensor->exposure,
slope_table1,
motor->steps1,
move, motor->fwdbwd, &z1, &z2);
/* no z1/z2 for sheetfed scanners */
if (dev->model->is_sheetfed == SANE_TRUE)
{
z1 = 0;
z2 = 0;
}
gl646_set_double_reg (regs, REG_Z1MOD, z1);
gl646_set_double_reg (regs, REG_Z2MOD, z2);
regs[reg_0x6b].value = motor->steps2;
regs[reg_0x6c].value =
(regs[reg_0x6c].value & REG6C_TGTIME) | ((z1 >> 13) & 0x38) | ((z2 >> 16)
& 0x07);
RIE (write_control (dev, xresolution));
/* setup analog frontend */
RIE (gl646_set_fe (dev, AFE_SET, xresolution));
/* now we're done with registers setup values used by data transfer */
/* we setup values needed for the data transfer */
/* we must use a round number of words_per_line */
/* CCD scanners have different requirements than CIS, and sheetfed ones
* need the buffer to be short enough not to miss the end of document in the feeder*/
if (dev->model->is_sheetfed == SANE_FALSE)
{
requested_buffer_size = (GL646_BULKIN_MAXSIZE / wpl) * wpl;
/* TODO seems CIS scanners should be treated differently */
read_buffer_size =
2 * requested_buffer_size +
((max_shift + stagger) * scan_settings.pixels * channels * depth) / 8;
}
else
{
requested_buffer_size = 8 * wpl * channels;
read_buffer_size = requested_buffer_size;
}
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 * scan_settings.pixels * channels *
depth) / 8));
/* scan bytes to read */
dev->read_bytes_left = wpl * linecnt;
DBG (DBG_info,
"gl646_setup_registers: physical bytes to read = %lu\n",
(u_long) dev->read_bytes_left);
dev->read_active = SANE_TRUE;
dev->current_setup.pixels =
((endx - startx) * sensor->xdpi) / dev->sensor.optical_res;
dev->current_setup.lines = linecnt;
dev->current_setup.depth = depth;
dev->current_setup.channels = channels;
dev->current_setup.exposure_time = sensor->exposure;
dev->current_setup.xres = sensor->xdpi;
dev->current_setup.yres = motor->ydpi;
dev->current_setup.half_ccd = half_ccd;
dev->current_setup.stagger = stagger;
dev->current_setup.max_shift = max_shift + stagger;
/* total_bytes_to_read is the number of byte to send to frontend
* total_bytes_read is the number of bytes sent to frontend
* read_bytes_left is the number of bytes to read from the scanner
*/
dev->total_bytes_read = 0;
if (depth == 1 || scan_settings.scan_mode == SCAN_MODE_LINEART)
dev->total_bytes_to_read =
((scan_settings.pixels * scan_settings.lines) / 8 +
(((scan_settings.pixels * scan_settings.lines) % 8) ? 1 : 0)) *
channels;
else
dev->total_bytes_to_read =
scan_settings.pixels * scan_settings.lines * channels * (depth / 8);
DBG (DBG_proc, "gl646_setup_registers: end\n");
return SANE_STATUS_GOOD;
}
/** copy sensor specific settings */
/* *dev : device infos
*regs : regiters to be set
extended : do extended set up
half_ccd: set up for half ccd resolution
all registers 08-0B, 10-1D, 52-5E are set up. They shouldn't
appear anywhere else but in register init
*/
static void
gl646_setup_sensor (Genesys_Device * dev, Genesys_Register_Set * regs)
{
Genesys_Register_Set *r;
int i;
DBG (DBG_proc, "gl646_setup_sensor: start\n");
for (i = 0; i < 4; i++)
{
r = sanei_genesys_get_address (regs, 0x08 + i);
r->value = dev->sensor.regs_0x08_0x0b[i];
}
for (i = 0; i < 14; i++)
{
r = sanei_genesys_get_address (regs, 0x10 + i);
r->value = dev->sensor.regs_0x10_0x1d[i];
}
for (i = 0; i < 13; i++)
{
r = sanei_genesys_get_address (regs, 0x52 + i);
r->value = dev->sensor.regs_0x52_0x5e[i];
}
DBG (DBG_proc, "gl646_setup_sensor: end\n");
}
/** Test if the ASIC works
*/
static SANE_Status
gl646_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, "gl646_asic_test: start\n");
/* 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, "gl646_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, "gl646_asic_test: failed to write register: %s\n",
sane_strstatus (status));
return status;
}
status = sanei_genesys_read_register (dev, 0x4e, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_asic_test: failed to read register: %s\n",
sane_strstatus (status));
return status;
}
if (val != 0xde) /* value of register 0x38 */
{
DBG (DBG_error, "gl646_asic_test: register contains invalid value\n");
return SANE_STATUS_IO_ERROR;
}
status = sanei_genesys_read_register (dev, 0x4f, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_asic_test: failed to read register: %s\n",
sane_strstatus (status));
return status;
}
if (val != 0xad) /* value of register 0x39 */
{
DBG (DBG_error, "gl646_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, "gl646_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, "gl646_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, "gl646_asic_test: failed to set buffer address: %s\n",
sane_strstatus (status));
free (data);
free (verify_data);
return status;
}
status = gl646_bulk_write_data (dev, 0x3c, data, size);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_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, "gl646_asic_test: failed to set buffer address: %s\n",
sane_strstatus (status));
free (data);
free (verify_data);
return status;
}
status =
gl646_bulk_read_data (dev, 0x45, (uint8_t *) verify_data, verify_size);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_asic_test: failed to bulk read data: %s\n",
sane_strstatus (status));
free (data);
free (verify_data);
return status;
}
/* i + 2 is needed as the changed address goes into effect only after one
data word is sent. */
for (i = 0; i < size; i++)
{
if (verify_data[i + 2] != data[i])
{
DBG (DBG_error, "gl646_asic_test: data verification error\n");
free (data);
free (verify_data);
return SANE_STATUS_IO_ERROR;
}
}
free (data);
free (verify_data);
DBG (DBG_info, "gl646_asic_test: end\n");
return SANE_STATUS_GOOD;
}
/* returns the max register bulk size */
static int
gl646_bulk_full_size (void)
{
return GENESYS_GL646_MAX_REGS;
}
/**
* Set all registers to default values after init
* @param dev scannerr's device to set
*/
static void
gl646_init_regs (Genesys_Device * dev)
{
int nr, addr;
DBG (DBG_proc, "gl646_init_regs\n");
nr = 0;
memset (dev->reg, 0, GENESYS_MAX_REGS * sizeof (Genesys_Register_Set));
for (addr = 1; addr <= 0x0b; addr++)
dev->reg[nr++].address = addr;
for (addr = 0x10; addr <= 0x29; addr++)
dev->reg[nr++].address = addr;
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 <= 0x5e; addr++)
dev->reg[nr++].address = addr;
for (addr = 0x60; addr <= 0x6d; addr++)
dev->reg[nr++].address = addr;
dev->reg[reg_0x01].value = 0x20 /*0x22 */ ; /* enable shading, CCD, color, 1M */
dev->reg[reg_0x02].value = 0x30 /*0x38 */ ; /* auto home, one-table-move, full step */
if (dev->model->motor_type == MOTOR_5345)
dev->reg[reg_0x02].value |= 0x01; /* half-step */
switch (dev->model->motor_type)
{
case MOTOR_5345:
dev->reg[reg_0x02].value |= 0x01; /* half-step */
break;
case MOTOR_XP200:
/* for this sheetfed scanner, no AGOHOME, nor backtracking */
dev->reg[reg_0x02].value = 0x50;
break;
default:
break;
}
dev->reg[reg_0x03].value = 0x1f /*0x17 */ ; /* lamp on */
dev->reg[reg_0x04].value = 0x13 /*0x03 */ ; /* 8 bits data, 16 bits A/D, color, Wolfson fe *//* todo: according to spec, 0x0 is reserved? */
switch (dev->model->dac_type)
{
case DAC_AD_XP200:
dev->reg[reg_0x04].value = 0x12;
break;
default:
/* Wolfson frontend */
dev->reg[reg_0x04].value = 0x13;
break;
}
dev->reg[reg_0x05].value = 0x00; /* 12 bits gamma, disable gamma, 24 clocks/pixel */
switch (dev->sensor.optical_res)
{
case 600:
dev->reg[reg_0x05].value |= REG05_DPIHW_600;
break;
case 1200:
dev->reg[reg_0x05].value |= REG05_DPIHW_1200;
break;
case 2400:
dev->reg[reg_0x05].value |= REG05_DPIHW_2400;
break;
default:
dev->reg[reg_0x05].value |= REG05_DPIHW;
break;
}
if (dev->model->flags & GENESYS_FLAG_14BIT_GAMMA)
dev->reg[reg_0x05].value |= REG05_GMM14BIT;
if (dev->model->dac_type == DAC_AD_XP200)
dev->reg[reg_0x05].value |= 0x01; /* 12 clocks/pixel */
if (dev->model->ccd_type == CCD_HP2300)
dev->reg[reg_0x06].value = 0x00; /* PWRBIT off, shading gain=4, normal AFE image capture */
else
dev->reg[reg_0x06].value = 0x18; /* PWRBIT on, shading gain=8, normal AFE image capture */
gl646_setup_sensor (dev, dev->reg);
dev->reg[reg_0x1e].value = 0xf0; /* watch-dog time */
switch (dev->model->ccd_type)
{
case CCD_HP2300:
dev->reg[reg_0x1e].value = 0xf0;
dev->reg[reg_0x1f].value = 0x10;
dev->reg[reg_0x20].value = 0x20;
break;
case CCD_HP2400:
dev->reg[reg_0x1e].value = 0x80;
dev->reg[reg_0x1f].value = 0x10;
dev->reg[reg_0x20].value = 0x20;
break;
case CCD_HP3670:
dev->reg[reg_0x19].value = 0x2a;
dev->reg[reg_0x1e].value = 0x80;
dev->reg[reg_0x1f].value = 0x10;
dev->reg[reg_0x20].value = 0x20;
break;
case CIS_XP200:
dev->reg[reg_0x1e].value = 0x10;
dev->reg[reg_0x1f].value = 0x01;
dev->reg[reg_0x20].value = 0x50;
break;
default:
dev->reg[reg_0x1f].value = 0x01;
dev->reg[reg_0x20].value = 0x50;
break;
}
dev->reg[reg_0x21].value = 0x08 /*0x20 */ ; /* table one steps number for forward slope curve of the acc/dec */
dev->reg[reg_0x22].value = 0x10 /*0x08 */ ; /* steps number of the forward steps for start/stop */
dev->reg[reg_0x23].value = 0x10 /*0x08 */ ; /* steps number of the backward steps for start/stop */
dev->reg[reg_0x24].value = 0x08 /*0x20 */ ; /* table one steps number backward slope curve of the acc/dec */
dev->reg[reg_0x25].value = 0x00; /* scan line numbers (7000) */
dev->reg[reg_0x26].value = 0x00 /*0x1b */ ;
dev->reg[reg_0x27].value = 0xd4 /*0x58 */ ;
dev->reg[reg_0x28].value = 0x01; /* PWM duty for lamp control */
dev->reg[reg_0x29].value = 0xff;
dev->reg[reg_0x2c].value = 0x02; /* set resolution (600 DPI) */
dev->reg[reg_0x2d].value = 0x58;
dev->reg[reg_0x2e].value = 0x78; /* set black&white threshold high level */
dev->reg[reg_0x2f].value = 0x7f; /* set black&white threshold low level */
dev->reg[reg_0x30].value = 0x00; /* begin pixel position (16) */
dev->reg[reg_0x31].value = dev->sensor.dummy_pixel /*0x10 */ ; /* TGW + 2*TG_SHLD + x */
dev->reg[reg_0x32].value = 0x2a /*0x15 */ ; /* end pixel position (5390) */
dev->reg[reg_0x33].value = 0xf8 /*0x0e */ ; /* TGW + 2*TG_SHLD + y */
dev->reg[reg_0x34].value = dev->sensor.dummy_pixel;
dev->reg[reg_0x35].value = 0x01 /*0x00 */ ; /* set maximum word size per line, for buffer full control (10800) */
dev->reg[reg_0x36].value = 0x00 /*0x2a */ ;
dev->reg[reg_0x37].value = 0x00 /*0x30 */ ;
dev->reg[reg_0x38].value = HIBYTE (dev->settings.exposure_time) /*0x2a */ ; /* line period (exposure time = 11000 pixels) */
dev->reg[reg_0x39].value = LOBYTE (dev->settings.exposure_time) /*0xf8 */ ;
dev->reg[reg_0x3d].value = 0x00; /* set feed steps number of motor move */
dev->reg[reg_0x3e].value = 0x00;
dev->reg[reg_0x3f].value = 0x01 /*0x00 */ ;
dev->reg[reg_0x60].value = 0x00; /* Z1MOD, 60h:61h:(6D b5:b3), remainder for start/stop */
dev->reg[reg_0x61].value = 0x00; /* (21h+22h)/LPeriod */
dev->reg[reg_0x62].value = 0x00; /* Z2MODE, 62h:63h:(6D b2:b0), remainder for start scan */
dev->reg[reg_0x63].value = 0x00; /* (3Dh+3Eh+3Fh)/LPeriod for one-table mode,(21h+1Fh)/LPeriod */
dev->reg[reg_0x64].value = 0x00; /* motor PWM frequency */
dev->reg[reg_0x65].value = 0x00; /* PWM duty cycle for table one motor phase (63 = max) */
if (dev->model->motor_type == MOTOR_5345)
dev->reg[reg_0x65].value = 0x02; /* PWM duty cycle for table one motor phase (63 = max) */
dev->reg[reg_0x66].value = dev->gpo.value[0];
dev->reg[reg_0x67].value = dev->gpo.value[1];
dev->reg[reg_0x68].value = dev->gpo.enable[0];
dev->reg[reg_0x69].value = dev->gpo.enable[1];
switch (dev->model->motor_type)
{
case MOTOR_HP2300:
case MOTOR_HP2400:
dev->reg[reg_0x6a].value = 0x7f; /* table two steps number for acc/dec */
dev->reg[reg_0x6b].value = 0x78; /* table two steps number for acc/dec */
dev->reg[reg_0x6d].value = 0x7f;
break;
case MOTOR_5345:
dev->reg[reg_0x6a].value = 0x42; /* table two fast moving step type, PWM duty for table two */
dev->reg[reg_0x6b].value = 0xff; /* table two steps number for acc/dec */
dev->reg[reg_0x6d].value = 0x41; /* select deceleration steps whenever go home (0), accel/decel stop time (31 * LPeriod) */
break;
case MOTOR_XP200:
dev->reg[reg_0x6a].value = 0x7f; /* table two fast moving step type, PWM duty for table two */
dev->reg[reg_0x6b].value = 0x08; /* table two steps number for acc/dec */
dev->reg[reg_0x6d].value = 0x01; /* select deceleration steps whenever go home (0), accel/decel stop time (31 * LPeriod) */
break;
case MOTOR_HP3670:
dev->reg[reg_0x6a].value = 0x41; /* table two steps number for acc/dec */
dev->reg[reg_0x6b].value = 0xc8; /* table two steps number for acc/dec */
dev->reg[reg_0x6d].value = 0x7f;
break;
default:
dev->reg[reg_0x6a].value = 0x40; /* table two fast moving step type, PWM duty for table two */
dev->reg[reg_0x6b].value = 0xff; /* table two steps number for acc/dec */
dev->reg[reg_0x6d].value = 0x01; /* select deceleration steps whenever go home (0), accel/decel stop time (31 * LPeriod) */
break;
}
dev->reg[reg_0x6c].value = 0x00; /* peroid times for LPeriod, expR,expG,expB, Z1MODE, Z2MODE (one period time) */
}
/* Send slope table for motor movement
slope_table in machine byte order
*/
static SANE_Status
gl646_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;
#ifdef WORDS_BIGENDIAN
int i;
#endif
DBG (DBG_proc,
"gl646_send_slope_table (table_nr = %d, steps = %d)=%d .. %d\n",
table_nr, steps, slope_table[0], slope_table[steps - 1]);
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 = 0x1f800;
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
status =
sanei_genesys_set_buffer_address (dev, start_address + table_nr * 0x100);
if (status != SANE_STATUS_GOOD)
{
#ifdef WORDS_BIGENDIAN
free (table);
#endif
DBG (DBG_error,
"gl646_send_slope_table: failed to set buffer address: %s\n",
sane_strstatus (status));
return status;
}
status = gl646_bulk_write_data (dev, 0x3c, (uint8_t *) table, steps * 2);
if (status != SANE_STATUS_GOOD)
{
#ifdef WORDS_BIGENDIAN
free (table);
#endif
DBG (DBG_error,
"gl646_send_slope_table: failed to send slope table: %s\n",
sane_strstatus (status));
return status;
}
#ifdef WORDS_BIGENDIAN
free (table);
#endif
DBG (DBG_proc, "gl646_send_slope_table: end\n");
return status;
}
/* Set values of Analog Device type frontend */
static SANE_Status
gl646_set_ad_fe (Genesys_Device * dev, uint8_t set)
{
SANE_Status status = SANE_STATUS_GOOD;
int i;
uint16_t val;
DBG (DBG_proc, "gl646_set_ad_fe(): start\n");
if (set == AFE_INIT)
{
DBG (DBG_proc, "gl646_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 */
val = dev->frontend.reg[0];
status = sanei_genesys_fe_write_data (dev, 0x00, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_ad_fe: failed to write reg0: %s\n",
sane_strstatus (status));
return status;
}
val = dev->frontend.reg[1];
status = sanei_genesys_fe_write_data (dev, 0x01, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_ad_fe: failed to write reg1: %s\n",
sane_strstatus (status));
return status;
}
}
if (set == AFE_SET)
{
for (i = 0; i < 3; i++)
{
val = dev->frontend.gain[i];
status = sanei_genesys_fe_write_data (dev, 0x02 + i, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_set_ad_fe: failed to write gain %d: %s\n", i,
sane_strstatus (status));
return status;
}
}
for (i = 0; i < 3; i++)
{
val = dev->frontend.offset[i];
status = sanei_genesys_fe_write_data (dev, 0x05 + i, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_set_ad_fe: failed to write offset %d: %s\n", i,
sane_strstatus (status));
return status;
}
}
}
/*
if (set == AFE_POWER_SAVE)
{
status =
sanei_genesys_fe_write_data (dev, 0x00, dev->frontend.reg[0] | 0x04);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_ad_fe: failed to write reg0: %s\n",
sane_strstatus (status));
return status;
}
} */
DBG (DBG_proc, "gl646_set_ad_fe(): end\n");
return status;
}
/** set up analog frontend
* set up analog frontend
* @param dev device to set up
* @param set action from AFE_SET, AFE_INIT and AFE_POWERSAVE
* @param dpi resolution of the scan since it affects settings
* @return SANE_STATUS_GOOD if evrithing OK
*/
static SANE_Status
gl646_wm_hp3670 (Genesys_Device * dev, uint8_t set, int dpi)
{
SANE_Status status = SANE_STATUS_GOOD;
int i;
DBG (DBG_proc, "gl646_wm_hp3670: start \n");
switch (set)
{
case AFE_INIT:
status = sanei_genesys_fe_write_data (dev, 0x04, 0x80);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: reset failed: %s\n",
sane_strstatus (status));
return status;
}
usleep (200000UL);
RIE (sanei_genesys_write_register (dev, 0x50, 0x00));
sanei_genesys_init_fe (dev);
status = sanei_genesys_fe_write_data (dev, 0x01, dev->frontend.reg[1]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing reg1 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, "gl646_wm_hp3670: writing reg2 failed: %s\n",
sane_strstatus (status));
return status;
}
status = gl646_gpio_output_enable (dev->dn, 0x07);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: failed to enable GPIO: %s\n",
sane_strstatus (status));
return status;
}
break;
case AFE_POWER_SAVE:
status = sanei_genesys_fe_write_data (dev, 0x01, 0x06);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing reg1 failed: %s\n",
sane_strstatus (status));
return status;
}
status = sanei_genesys_fe_write_data (dev, 0x06, 0x0f);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing reg6 failed: %s\n",
sane_strstatus (status));
return status;
}
return status;
break;
default: /* AFE_SET */
/* mode setup */
i = dev->frontend.reg[3];
if (dpi > dev->sensor.optical_res / 2)
{
i = i & 0x1f;
}
status = sanei_genesys_fe_write_data (dev, 0x03, i);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing reg3 failed: %s\n",
sane_strstatus (status));
return status;
}
/* offset and sign (or msb/lsb ?) */
for (i = 0; i < 3; i++)
{
status =
sanei_genesys_fe_write_data (dev, 0x20 + i,
dev->frontend.offset[i]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_wm_hp3670: writing offset%d failed: %s\n", i,
sane_strstatus (status));
return status;
}
status = sanei_genesys_fe_write_data (dev, 0x24 + i, dev->frontend.sign[i]); /* MSB/LSB ? */
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing sign%d failed: %s\n",
i, sane_strstatus (status));
return status;
}
}
/* gain */
for (i = 0; i < 3; i++)
{
status =
sanei_genesys_fe_write_data (dev, 0x28 + i,
dev->frontend.gain[i]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_wm_hp3670: writing gain%d failed: %s\n",
i, sane_strstatus (status));
return status;
}
}
}
DBG (DBG_proc, "gl646_wm_hp3670: success \n");
return status;
}
/** Set values of analog frontend
* @param dev device to set
* @param set action to execute
* @param dpi dpi to setup the AFE
* @return error or SANE_STATUS_GOOD */
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_set_fe (Genesys_Device * dev, uint8_t set, int dpi)
{
SANE_Status status;
int i;
uint8_t val;
DBG (DBG_proc, "gl646_set_fe (%s,%d)\n",
set == AFE_INIT ? "init" : set == AFE_SET ? "set" : set ==
AFE_POWER_SAVE ? "powersave" : "huh?", dpi);
/* Analog Device type frontend */
if ((dev->reg[reg_0x04].value & REG04_FESET) == 0x02)
return gl646_set_ad_fe (dev, set);
/* Wolfson type frontend */
if ((dev->reg[reg_0x04].value & REG04_FESET) != 0x03)
{
DBG (DBG_proc, "gl646_set_fe(): unspported frontend type %d\n",
dev->reg[reg_0x04].value & REG04_FESET);
return SANE_STATUS_UNSUPPORTED;
}
/* per frontend function to keep code clean */
switch (dev->model->dac_type)
{
case DAC_WOLFSON_HP3670:
case DAC_WOLFSON_HP2400:
return gl646_wm_hp3670 (dev, set, dpi);
break;
default:
DBG (DBG_proc, "gl646_set_fe(): using old method\n");
break;
}
/* initialize analog frontend */
if (set == AFE_INIT)
{
DBG (DBG_proc, "gl646_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, "gl646_set_fe: init fe failed: %s\n",
sane_strstatus (status));
return status;
}
/* enable GPIO for some models */
if (dev->model->ccd_type == CCD_HP2300)
{
val = 0x07;
status = gl646_gpio_output_enable (dev->dn, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_set_fe: failed to enable GPIO: %s\n",
sane_strstatus (status));
return status;
}
}
return status;
}
/* set fontend to power saving mode */
if (set == AFE_POWER_SAVE)
{
status = sanei_genesys_fe_write_data (dev, 0x01, 0x02);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_fe: writing data failed: %s\n",
sane_strstatus (status));
}
return status;
}
/* here starts AFE_SET */
/* 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_HP3670
&& 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, "gl646_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, "gl646_set_fe: writing reg2 failed: %s\n",
sane_strstatus (status));
return status;
}
}
/* start with reg3 */
status = sanei_genesys_fe_write_data (dev, 0x03, dev->frontend.reg[3]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_fe: writing reg3 failed: %s\n",
sane_strstatus (status));
return status;
}
switch (dev->model->ccd_type)
{
default:
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, "gl646_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, "gl646_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,
"gl646_set_fe: writing offset[%d] failed: %s\n", i,
sane_strstatus (status));
return status;
}
}
break;
/* just can't have it to work ....
case CCD_HP2300:
case CCD_HP2400:
case CCD_HP3670:
status =
sanei_genesys_fe_write_data (dev, 0x23, dev->frontend.offset[1]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_set_fe: writing offset[1] failed: %s\n",
sane_strstatus (status));
return status;
}
status = sanei_genesys_fe_write_data (dev, 0x28, dev->frontend.gain[1]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_fe: writing gain[1] failed: %s\n",
sane_strstatus (status));
return status;
}
break; */
}
/* end with reg1 */
status = sanei_genesys_fe_write_data (dev, 0x01, dev->frontend.reg[1]);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_set_fe: writing reg1 failed: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_proc, "gl646_set_fe: end\n");
return SANE_STATUS_GOOD;
}
/** Set values of analog frontend
* this this the public interface, the gl646 as to use one more
* parameter to work effectively, hence the redirection
* @param dev device to set
* @param set action to execute
* @return error or SANE_STATUS_GOOD */
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_public_set_fe (Genesys_Device * dev, uint8_t set)
{
return gl646_set_fe (dev, set, dev->settings.yres);
}
static void
gl646_set_motor_power (Genesys_Register_Set * regs, SANE_Bool set)
{
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
gl646_set_lamp_power (Genesys_Device * dev,
Genesys_Register_Set * regs, SANE_Bool set)
{
if (dev)
{
if (set)
{
sanei_genesys_set_reg_from_set (regs, 0x03,
sanei_genesys_read_reg_from_set
(regs, 0x03) | REG03_LAMPPWR);
}
else
{
sanei_genesys_set_reg_from_set (regs, 0x03,
sanei_genesys_read_reg_from_set
(regs, 0x03) & ~REG03_LAMPPWR);
}
}
}
/**
* enters or leaves power saving mode
* limited to AFE for now.
* @param dev scanner's device
* @param SANE_TRUE to enable power saving, SANE_FALSE to leave it
* @return allways SANE_STATUS_GOOD
*/
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_save_power (Genesys_Device * dev, SANE_Bool enable)
{
DBG (DBG_proc, "gl646_save_power: start\n");
DBG (DBG_info, "gl646_save_power: enable = %d\n", enable);
if (enable)
{
/* gl646_set_fe (dev, AFE_POWER_SAVE); */
}
else
{
gl646_set_fe (dev, AFE_INIT, 0);
}
DBG (DBG_proc, "gl646_save_power: end\n");
return SANE_STATUS_GOOD;
}
static SANE_Status
gl646_set_powersaving (Genesys_Device * dev, int delay /* in minutes */ )
{
SANE_Status status = SANE_STATUS_GOOD;
Genesys_Register_Set local_reg[6];
int rate, exposure_time, tgtime, time;
DBG (DBG_proc, "gl646_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 = 0x38; /* line period low */
local_reg[3].value = 0x00;
local_reg[4].address = 0x39; /* line period high */
local_reg[4].value = 0x00;
local_reg[5].address = 0x6c; /* period times for LPeriod, expR,expG,expB, Z1MODE, Z2MODE */
local_reg[5].value = 0x00;
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[5].value |= tgtime << 6;
exposure_time /= rate;
if (exposure_time > 65535)
exposure_time = 65535;
local_reg[3].value = exposure_time / 256; /* highbyte */
local_reg[4].value = exposure_time & 255; /* lowbyte */
status = gl646_bulk_write_register (dev, local_reg,
sizeof (local_reg) /
sizeof (local_reg[0]));
if (status != SANE_STATUS_GOOD)
DBG (DBG_error,
"gl646_set_powersaving: Failed to bulk write registers: %s\n",
sane_strstatus (status));
DBG (DBG_proc, "gl646_set_powersaving: end\n");
return status;
}
/**
* loads document into scanner
* currently only used by XP200
* bit2 (0x04) of gpio is paper event (document in/out) on XP200
* HOMESNR is set if no document in front of sensor, the sequence of events is
* paper event -> document is in the sheet feeder
* HOMESNR becomes 0 -> document reach sensor
* HOMESNR becomes 1 ->document left sensor
* paper event -> document is out
*/
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_load_document (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
Genesys_Register_Set regs[11];
unsigned int used, vfinal, count;
uint16_t slope_table[255];
uint8_t val;
DBG (DBG_proc, "gl646_load_document: start\n");
/* no need to load document is flatbed scanner */
if (dev->model->is_sheetfed == SANE_FALSE)
{
DBG (DBG_proc, "gl646_load_document: nothing to load\n");
DBG (DBG_proc, "gl646_load_document: end\n");
return SANE_STATUS_GOOD;
}
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to read status: %s\n",
sane_strstatus (status));
return status;
}
/* HOMSNR is set if a document is inserted */
if ((val & REG41_HOMESNR))
{
/* if no document, waits for a paper event to start loading */
/* with a 60 seconde minutes timeout */
count = 0;
do
{
status = gl646_gpio_read (dev->dn, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to read paper sensor %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_info, "gl646_load_document: GPIO=0x%02x\n", val);
if ((val & 0x04) != 0x04)
{
DBG (DBG_warn, "gl646_load_document: no paper detected\n");
}
usleep (200000UL); /* sleep 200 ms */
count++;
}
while (((val & 0x04) != 0x04) && (count < 300)); /* 1 min time out */
if (count == 300)
{
DBG (DBG_error,
"gl646_load_document: timeout waiting for document\n");
return SANE_STATUS_NO_DOCS;
}
}
/* set up to fast move before scan then move until document is detected */
regs[0].address = 0x01;
regs[0].value = 0x90;
/* AGOME, 2 slopes motor moving */
regs[1].address = 0x02;
regs[1].value = 0x79;
/* motor feeding steps to 0 */
regs[2].address = 0x3d;
regs[2].value = 0;
regs[3].address = 0x3e;
regs[3].value = 0;
regs[4].address = 0x3f;
regs[4].value = 0;
/* 50 fast moving steps */
regs[5].address = 0x6b;
regs[5].value = 50;
/* set GPO */
regs[6].address = 0x66;
regs[6].value = 0x30;
/* stesp NO */
regs[7].address = 0x21;
regs[7].value = 4;
regs[8].address = 0x22;
regs[8].value = 1;
regs[9].address = 0x23;
regs[9].value = 1;
regs[10].address = 0x24;
regs[10].value = 4;
/* generate slope table 2 */
sanei_genesys_generate_slope_table (slope_table,
50,
51,
2400,
6000, 2400, 50, 0.25, &used, &vfinal);
/* document loading:
* send regs
* start motor
* wait e1 status to become e0
*/
status = gl646_send_slope_table (dev, 1, slope_table, 50);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to send slope table 1: %s\n",
sane_strstatus (status));
return status;
}
status =
gl646_bulk_write_register (dev, regs, sizeof (regs) / sizeof (regs[0]));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to bulk write registers: %s\n",
sane_strstatus (status));
return status;
}
status = gl646_start_motor (dev);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to start motor: %s\n",
sane_strstatus (status));
return SANE_STATUS_IO_ERROR;
}
count = 0;
do
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to read status: %s\n",
sane_strstatus (status));
return status;
}
usleep (200000UL); /* sleep 200 ms */
count++;
}
while ((val & REG41_MOTMFLG) && (count < 300));
if (count == 300)
{
DBG (DBG_error, "gl646_load_document: can't load document\n");
return SANE_STATUS_JAMMED;
}
/* when loading OK, document is here */
dev->document = SANE_TRUE;
/* set up to idle */
regs[1].value = 0x71;
regs[4].value = 1;
regs[5].value = 8;
status =
gl646_bulk_write_register (dev, regs, sizeof (regs) / sizeof (regs[0]));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_load_document: failed to bulk write idle registers: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_proc, "gl646_load_document: end\n");
return status;
}
/**
* detects end of document and adjust current scan
* to take it into account
* used by sheetfed scanners
*/
static SANE_Status
gl646_detect_document_end (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
uint8_t val, gpio;
unsigned int bytes_left, lines;
DBG (DBG_proc, "gl646_detect_document_end: start\n");
/* test for document presence */
RIE (sanei_genesys_get_status (dev, &val));
if (DBG_LEVEL > DBG_info)
{
print_status (val);
}
status = gl646_gpio_read (dev->dn, &gpio);
DBG (DBG_info, "gl646_detect_document_end: GPIO=0x%02x\n", gpio);
/* detect document event. There one event when the document go in,
* then another when it leaves */
if ((dev->document == SANE_TRUE) && (gpio & 0x04)
&& (dev->total_bytes_read > 0))
{
DBG (DBG_info, "gl646_detect_document_end: no more document\n");
dev->document = SANE_FALSE;
/* adjust number of bytes to read:
* total_bytes_to_read is the number of byte to send to frontend
* total_bytes_read is the number of bytes sent to frontend
* read_bytes_left is the number of bytes to read from the scanner
*/
DBG (DBG_io, "gl646_detect_document_end: total_bytes_to_read=%lu\n",
(u_long) dev->total_bytes_to_read);
DBG (DBG_io, "gl646_detect_document_end: total_bytes_read =%lu\n",
(u_long) dev->total_bytes_read);
DBG (DBG_io, "gl646_detect_document_end: read_bytes_left =%lu\n",
(u_long) dev->read_bytes_left);
/* amount of data available from scanner is what to scan */
status = sanei_genesys_read_valid_words (dev, &bytes_left);
/* we add the number of lines needed to read the last part of the document in */
lines =
(SANE_UNFIX (dev->model->y_offset) * dev->current_setup.yres) /
MM_PER_INCH;
DBG (DBG_io, "gl646_detect_document_end: adding %d line to flush\n",
lines);
bytes_left += lines * dev->wpl;
if (dev->current_setup.depth > 8)
{
bytes_left = 2 * bytes_left;
}
if (dev->current_setup.channels > 1)
{
bytes_left = 3 * bytes_left;
}
if (bytes_left < dev->read_bytes_left)
{
dev->total_bytes_to_read = dev->total_bytes_read + bytes_left;
dev->read_bytes_left = bytes_left;
}
DBG (DBG_io, "gl646_detect_document_end: total_bytes_to_read=%lu\n",
(u_long) dev->total_bytes_to_read);
DBG (DBG_io, "gl646_detect_document_end: total_bytes_read =%lu\n",
(u_long) dev->total_bytes_read);
DBG (DBG_io, "gl646_detect_document_end: read_bytes_left =%lu\n",
(u_long) dev->read_bytes_left);
}
DBG (DBG_proc, "gl646_detect_document_end: end\n");
return status;
}
/**
* eject document from the feeder
* currently only used by XP200
* TODO we currently rely on AGOHOME not being set for sheetfed scanners,
* maybe check this flag in eject to let the document being eject automaticaly
*/
static SANE_Status
gl646_eject_document (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
Genesys_Register_Set regs[11];
unsigned int used, vfinal, count;
uint16_t slope_table[255];
uint8_t gpio, state;
DBG (DBG_proc, "gl646_eject_document: start\n");
/* at the end there will be noe more document */
dev->document = SANE_FALSE;
/* first check for document event */
status = gl646_gpio_read (dev->dn, &gpio);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to read paper sensor %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_info, "gl646_eject_document: GPIO=0x%02x\n", gpio);
/* test status : paper event + HOMESNR -> no more doc ? */
status = sanei_genesys_get_status (dev, &state);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to read status: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_info, "gl646_eject_document: state=0x%02x\n", state);
if (DBG_LEVEL > DBG_info)
{
print_status (state);
}
/* HOMSNR=0 if no document inserted */
if ((state & REG41_HOMESNR) != 0)
{
dev->document = SANE_FALSE;
DBG (DBG_info, "gl646_eject_document: no more document to eject\n");
DBG (DBG_proc, "gl646_eject_document: end\n");
return status;
}
/* there is a document inserted, eject it */
status = sanei_genesys_write_register (dev, 0x01, 0xb0);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to write register: %s\n",
sane_strstatus (status));
return status;
}
/* wait for motor to stop */
do
{
usleep (200000UL);
status = sanei_genesys_get_status (dev, &state);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to read status: %s\n",
sane_strstatus (status));
return status;
}
}
while (state & REG41_MOTMFLG);
/* set up to fast move before scan then move until document is detected */
regs[0].address = 0x01;
regs[0].value = 0xb0;
/* AGOME, 2 slopes motor moving , eject 'backward' */
regs[1].address = 0x02;
regs[1].value = 0x5d;
/* motor feeding steps to 119880 */
regs[2].address = 0x3d;
regs[2].value = 1;
regs[3].address = 0x3e;
regs[3].value = 0xd4;
regs[4].address = 0x3f;
regs[4].value = 0x48;
/* 60 fast moving steps */
regs[5].address = 0x6b;
regs[5].value = 60;
/* set GPO */
regs[6].address = 0x66;
regs[6].value = 0x30;
/* stesp NO */
regs[7].address = 0x21;
regs[7].value = 4;
regs[8].address = 0x22;
regs[8].value = 1;
regs[9].address = 0x23;
regs[9].value = 1;
regs[10].address = 0x24;
regs[10].value = 4;
/* generate slope table 2 */
sanei_genesys_generate_slope_table (slope_table,
60,
61,
1600,
10000, 1600, 60, 0.25, &used, &vfinal);
/* document eject:
* send regs
* start motor
* wait c1 status to become c8 : HOMESNR and ~MOTFLAG
*/
status = gl646_send_slope_table (dev, 1, slope_table, 60);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to send slope table 1: %s\n",
sane_strstatus (status));
return status;
}
status =
gl646_bulk_write_register (dev, regs, sizeof (regs) / sizeof (regs[0]));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to bulk write registers: %s\n",
sane_strstatus (status));
return status;
}
status = gl646_start_motor (dev);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to start motor: %s\n",
sane_strstatus (status));
return SANE_STATUS_IO_ERROR;
}
/* loop until paper sensor tells paper is out, and till motor is running */
/* use a 30 timeout */
count = 0;
do
{
status = sanei_genesys_get_status (dev, &state);
print_status (state);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to read status: %s\n",
sane_strstatus (status));
return status;
}
usleep (200000UL); /* sleep 200 ms */
count++;
}
while (((state & REG41_HOMESNR) == 0) && (count < 150));
/* read GPIO on exit */
status = gl646_gpio_read (dev->dn, &gpio);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_eject_document: failed to read paper sensor %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_info, "gl646_eject_document: GPIO=0x%02x\n", gpio);
DBG (DBG_proc, "gl646_eject_document: end\n");
return status;
}
/* Send the low-level scan command */
static SANE_Status
gl646_begin_scan (Genesys_Device * dev, Genesys_Register_Set * reg,
SANE_Bool start_motor)
{
SANE_Status status;
Genesys_Register_Set local_reg[3];
DBG (DBG_proc, "gl646_begin_scan\n");
local_reg[0].address = 0x03;
local_reg[0].value = sanei_genesys_read_reg_from_set (reg, 0x03);
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 = 0x0f;
if (start_motor)
local_reg[2].value = 0x01;
else
local_reg[2].value = 0x00; /* do not start motor yet */
status = gl646_bulk_write_register (dev, local_reg,
sizeof (local_reg) /
sizeof (local_reg[0]));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_begin_scan: failed to bulk write registers: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_proc, "gl646_begin_scan: end\n");
return status;
}
/* Send the stop scan command */
static SANE_Status
end_scan (Genesys_Device * dev, Genesys_Register_Set * reg,
SANE_Bool check_stop, SANE_Bool eject)
{
SANE_Status status = SANE_STATUS_GOOD;
int i = 0;
uint8_t val, scanfsh = 0;
DBG (DBG_proc, "end_scan (check_stop = %d, eject = %d)\n", check_stop,
eject);
/* we need to compute scanfsh before cancelling scan */
if (dev->model->is_sheetfed == SANE_TRUE)
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"end_scan: failed to read register: %s\n",
sane_strstatus (status));
return status;
}
if (val & REG41_SCANFSH)
scanfsh = 1;
if (DBG_LEVEL > DBG_io2)
{
print_status (val);
}
}
/* ends scan */
val = sanei_genesys_read_reg_from_set (reg, 0x01);
val &= ~REG01_SCAN;
sanei_genesys_set_reg_from_set (reg, 0x01, val);
status = sanei_genesys_write_register (dev, 0x01, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"end_scan: failed to write register 01: %s\n",
sane_strstatus (status));
return status;
}
/* for sheetfed scanners, we may have to eject document */
if (dev->model->is_sheetfed == SANE_TRUE)
{
if (eject == SANE_TRUE && dev->document == SANE_TRUE)
{
status = gl646_eject_document (dev);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "end_scan: failed to eject document\n");
return status;
}
}
if (check_stop)
{
for (i = 0; i < 30; i++) /* do not wait longer than wait 3 seconds */
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"end_scan: failed to read register: %s\n",
sane_strstatus (status));
return status;
}
if (val & REG41_SCANFSH)
scanfsh = 1;
if (DBG_LEVEL > DBG_io2)
{
print_status (val);
}
if (!(val & REG41_MOTMFLG) && (val & REG41_FEEDFSH) && scanfsh)
{
DBG (DBG_proc, "end_scan: scanfeed finished\n");
break; /* leave for loop */
}
usleep (10000UL); /* sleep 100 ms */
}
}
}
else /* flat bed scanners */
{
if (check_stop)
{
for (i = 0; i < 300; i++) /* do not wait longer than wait 30 seconds */
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"end_scan: failed to read register: %s\n",
sane_strstatus (status));
return status;
}
if (val & REG41_SCANFSH)
scanfsh = 1;
if (DBG_LEVEL > DBG_io)
{
print_status (val);
}
if (!(val & REG41_MOTMFLG) && (val & REG41_FEEDFSH) && scanfsh)
{
DBG (DBG_proc, "end_scan: scanfeed finished\n");
break; /* leave while loop */
}
if ((!(val & REG41_MOTMFLG)) && (val & REG41_HOMESNR))
{
DBG (DBG_proc, "end_scan: head at home\n");
break; /* leave while loop */
}
usleep (10000UL); /* sleep 100 ms */
}
}
}
DBG (DBG_proc, "end_scan: end (i=%u)\n", i);
return status;
}
/* Send the stop scan command */
static SANE_Status
gl646_end_scan (Genesys_Device * dev, Genesys_Register_Set * reg,
SANE_Bool check_stop)
{
return end_scan (dev, reg, check_stop, SANE_FALSE);
}
/**
* parks head
* @param dev scanner's device
* @param wait_until_home true if the function waits until head parked
*/
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_slow_back_home (Genesys_Device * dev, SANE_Bool wait_until_home)
{
SANE_Status status;
Genesys_Settings settings;
uint8_t val;
int i;
DBG (DBG_proc, "gl646_slow_back_home: start , wait_until_home = %d\n",
wait_until_home);
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_slow_back_home: failed to read home sensor: %s\n",
sane_strstatus (status));
return status;
}
if (DBG_LEVEL > DBG_io)
{
print_status (val);
}
dev->scanhead_position_in_steps = 0;
if (val & REG41_HOMESNR) /* is sensor at home? */
{
DBG (DBG_info, "gl646_slow_back_home: end since already at home\n");
return SANE_STATUS_GOOD;
}
/* stop motor if needed */
if (val & REG41_MOTMFLG)
{
status = gl646_stop_motor (dev);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_slow_back_home: failed to stop motor: %s\n",
sane_strstatus (status));
return SANE_STATUS_IO_ERROR;
}
usleep (200000UL);
}
/* when scanhead is moving then wait until scanhead stops or timeout */
DBG (DBG_info, "gl646_slow_back_home: ensuring that motor is off\n");
val = REG41_MOTMFLG;
for (i = 400; i > 0 && (val & REG41_MOTMFLG); i--) /* do not wait longer than 40 seconds, count down to get i = 0 when busy */
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_slow_back_home: Failed to read home sensor & motor status: %s\n",
sane_strstatus (status));
return status;
}
if (((val & (REG41_MOTMFLG | REG41_HOMESNR)) == REG41_HOMESNR)) /* at home and motor is off */
{
DBG (DBG_info,
"gl646_slow_back_home: already at home and not moving\n");
return SANE_STATUS_GOOD;
}
usleep (100 * 1000); /* sleep 100 ms (todo: fixed to really sleep 100 ms) */
}
if (!i) /* the loop counted down to 0, scanner still is busy */
{
DBG (DBG_error,
"gl646_slow_back_home: motor is still on: device busy\n");
return SANE_STATUS_DEVICE_BUSY;
}
/* setup for a backward scan of 65535 steps, with no actual data reading */
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres =
get_closest_resolution (dev->model->ccd_type, 75, SANE_FALSE);
settings.yres = settings.xres;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels = 600;
settings.lines = 1;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
status = setup_for_scan (dev, settings, SANE_TRUE, SANE_TRUE, SANE_TRUE);
/* backward , no actual data scanned TODO more setup flags to avoid this register manipulations ? */
dev->reg[reg_0x02].value |= REG02_MTRREV;
dev->reg[reg_0x01].value &= ~REG01_SCAN;
gl646_set_triple_reg (dev->reg, REG_FEEDL, 65535);
/* sets frontend */
status = gl646_set_fe (dev, AFE_SET, settings.xres);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_slow_back_home: failed to set frontend: %s\n",
sane_strstatus (status));
return status;
}
/* write scan registers */
status = gl646_bulk_write_register (dev, dev->reg,
sizeof (dev->reg) /
sizeof (dev->reg[0]));
if (status != SANE_STATUS_GOOD)
DBG (DBG_error,
"gl646_slow_back_home: failed to bulk write registers: %s\n",
sane_strstatus (status));
/* registers are restored to an iddl state, give up if no head to park */
if (dev->model->is_sheetfed == SANE_TRUE)
{
DBG (DBG_proc, "gl646_slow_back_home: end \n");
return SANE_STATUS_GOOD;
}
/* starts scan */
status = gl646_begin_scan (dev, dev->reg, SANE_TRUE);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_slow_back_home: failed to begin scan: \n");
return status;
}
/* loop until head parked */
if (wait_until_home)
{
int loop = 0;
while (loop < 3) /* do not wait longer then 30 seconds */
{
status = sanei_genesys_get_status (dev, &val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_slow_back_home: Failed to read home sensor: %s\n",
sane_strstatus (status));
return status;
}
if (val & 0x08) /* home sensor */
{
DBG (DBG_info, "gl646_slow_back_home: reached home position\n");
DBG (DBG_proc, "gl646_slow_back_home: end\n");
usleep (500000); /* sleep 500 ms before returning */
return SANE_STATUS_GOOD;
}
usleep (100000); /* sleep 100 ms */
}
/* when we come here then the scanner needed too much time for this, so we better stop the motor */
gl646_stop_motor (dev);
end_scan (dev, dev->reg, SANE_TRUE, SANE_FALSE);
DBG (DBG_error,
"gl646_slow_back_home: timeout while waiting for scanhead to go home\n");
return SANE_STATUS_IO_ERROR;
}
DBG (DBG_info, "gl646_slow_back_home: scanhead is still moving\n");
DBG (DBG_proc, "gl646_slow_back_home: end\n");
return SANE_STATUS_GOOD;
}
/**
* Automatically set top-left edge of the scan area by scanning an
* area at 300 dpi from very top of scanner
* @param dev device stucture describing the scanner
* @return SANE_STATUS_GOOD in cas of success, else failure code
*/
static SANE_Status
gl646_search_start_position (Genesys_Device * dev)
{
SANE_Status status;
unsigned char *data = NULL;
Genesys_Settings settings;
unsigned int resolution, x, y;
DBG (DBG_proc, "gl646_search_start_position: start\n");
/* we scan at 300 dpi */
resolution = get_closest_resolution (dev->model->ccd_type, 300, SANE_FALSE);
/* fill settings for a gray level scan */
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_GRAY;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels = 600;
settings.lines = dev->model->search_lines;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* scan the desired area */
status =
simple_scan (dev, settings, SANE_TRUE, SANE_TRUE, SANE_FALSE, &data);
/* process data if scan is OK */
if (status == SANE_STATUS_GOOD)
{
/* handle stagger case : reorder gray data and thus loose some lines */
if (dev->current_setup.stagger > 0)
{
DBG (DBG_proc, "gl646_search_start_position: 'un-staggering'\n");
for (y = 0; y < settings.lines - dev->current_setup.stagger; y++)
{
/* one point out of 2 is 'unaligned' */
for (x = 0; x < settings.pixels; x += 2)
{
data[y * settings.pixels + x] =
data[(y + dev->current_setup.stagger) * settings.pixels +
x];
}
}
/* correct line number */
settings.lines -= dev->current_setup.stagger;
}
if (DBG_LEVEL >= DBG_data)
{
sanei_genesys_write_pnm_file ("search_position.pnm",
data,
settings.depth,
1, settings.pixels, settings.lines);
}
}
else
{
DBG (DBG_error, "gl646_search_start_position: simple_scan failed\n");
free (data);
}
/* now search reference points on the data */
status =
sanei_genesys_search_reference_point (dev, data,
dev->sensor.CCD_start_xoffset,
resolution, settings.pixels,
settings.lines);
if (status != SANE_STATUS_GOOD)
{
free (data);
DBG (DBG_error,
"gl646_search_start_position: failed to set search reference point: %s\n",
sane_strstatus (status));
return status;
}
free (data);
DBG (DBG_proc, "gl646_search_start_position: end\n");
return status;
}
/**
* internally overriden during effective calibration
* sets up register for coarse gain calibration
*/
static SANE_Status
gl646_init_regs_for_coarse_calibration (Genesys_Device * dev)
{
DBG (DBG_proc, "gl646_init_regs_for_coarse_calibration\n");
DBG (DBG_proc, "gl646_init_register_for_coarse_calibration: end\n");
/* to make compilers happy ... */
if (!dev)
{
return SANE_STATUS_INVAL;
}
return SANE_STATUS_GOOD;
}
/**
* init registers for shading calibration
* we assume that scanner's head is on an area suiting shading calibration.
* We scan a full scan width area by the shading line number for the device
* at either at full sensor's resolution or half depending upon half_ccd
* @param dev scanner's device
* @return SANE_STATUS_GOOD if success, else error code
*/
static SANE_Status
gl646_init_regs_for_shading (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
Genesys_Settings settings;
/* 1: no half_ccd, 2: use half number of pixels */
int half_ccd = 1;
int cksel = 1;
DBG (DBG_proc, "gl646_init_register_for_shading: start\n");
/* when shading all (full width) line, we must adapt to half_ccd case */
if (dev->model->flags & GENESYS_FLAG_HALF_CCD_MODE)
{
/* walk the master mode list to find if half_ccd */
if (is_half_ccd (dev->model->ccd_type, dev->settings.xres, SANE_TRUE) ==
SANE_TRUE)
{
half_ccd = 2;
}
}
/* fill settings for scan : always a color scan */
settings.scan_method = dev->settings.scan_method;
settings.scan_mode = dev->settings.scan_mode;
if (dev->model->is_cis == SANE_FALSE)
{
settings.scan_mode = SCAN_MODE_COLOR;
}
settings.xres = dev->sensor.optical_res / half_ccd;
cksel = get_cksel (dev->model->ccd_type, dev->settings.xres, SANE_TRUE);
settings.xres = settings.xres / cksel;
settings.yres = settings.xres;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * settings.xres) / dev->sensor.optical_res;
settings.lines = dev->model->shading_lines * (3 - half_ccd);
settings.depth = 16;
settings.color_filter = dev->settings.color_filter;
settings.disable_interpolation = dev->settings.disable_interpolation;
settings.threshold = dev->settings.threshold;
settings.exposure_time = dev->settings.exposure_time;
/* keep account of the movement for final scan move */
dev->scanhead_position_in_steps += settings.lines;
/* we don't want top offset, but we need right margin to be the same
* than the one for the final scan */
status = setup_for_scan (dev, settings, SANE_TRUE, SANE_FALSE, SANE_FALSE);
/* used when sending shading calibration data */
dev->calib_pixels = settings.pixels;
dev->calib_channels = dev->current_setup.channels;
if (dev->model->is_cis == SANE_FALSE)
{
dev->calib_channels = 3;
}
/* no shading */
dev->reg[reg_0x01].value &= ~REG01_DVDSET;
dev->reg[reg_0x02].value |= REG02_ACDCDIS; /* ease backtracking */
dev->reg[reg_0x02].value &= ~(REG02_FASTFED | REG02_AGOHOME);
dev->reg[reg_0x05].value &= ~REG05_GMMENB;
gl646_set_motor_power (dev->reg, SANE_FALSE);
/* TODO another flag to setup regs ? */
/* enforce needed LINCNT, getting rid of extra lines for color reordering */
if (dev->model->is_cis == SANE_FALSE)
{
gl646_set_triple_reg (dev->reg, REG_LINCNT, dev->model->shading_lines);
}
else
{
gl646_set_triple_reg (dev->reg, REG_LINCNT,
dev->model->shading_lines * 3);
}
/* copy reg to calib_reg */
memcpy (dev->calib_reg, dev->reg,
GENESYS_GL646_MAX_REGS * sizeof (Genesys_Register_Set));
/* this is an hack to make calibration cache working .... */
/* if we don't do this, cache will be identified at the shading calibration
* dpi which is diferent from calibration one */
dev->current_setup.xres = dev->settings.xres;
DBG (DBG_info,
"gl646_init_register_for_shading:\n\tdev->settings.xres=%d\n\tdev->settings.yres=%d\n",
dev->settings.xres, dev->settings.yres);
DBG (DBG_proc, "gl646_init_register_for_shading: end\n");
return status;
}
/**
* set up registers for the actual scan. The scan's parameters are given
* through the device settings. It allocates the scan buffers.
*/
static SANE_Status
gl646_init_regs_for_scan (Genesys_Device * dev)
{
SANE_Status status;
/* park head after calibration if needed */
if (dev->scanhead_position_in_steps > 0
&& dev->settings.scan_method == SCAN_METHOD_FLATBED)
{
status = gl646_slow_back_home (dev, SANE_TRUE);
if (status != SANE_STATUS_GOOD)
{
return status;
}
dev->scanhead_position_in_steps = 0;
}
return setup_for_scan (dev, dev->settings, SANE_FALSE, SANE_TRUE,
SANE_TRUE);
}
/**
* set up registers for the actual scan. The scan's parameters are given
* through the device settings. It allocates the scan buffers.
* @param dev scanner's device
* @param settings settings of scan
* @param split SANE_TRUE if move to scan area is split from scan, SANE_FALSE is
* scan first moves to area
* @param xcorrection take x geometry correction into account (fixed and detected offsets)
* @param ycorrection take y geometry correction into account
*/
static SANE_Status
setup_for_scan (Genesys_Device * dev, Genesys_Settings settings,
SANE_Bool split, SANE_Bool xcorrection, SANE_Bool ycorrection)
{
SANE_Status status = SANE_STATUS_GOOD;
SANE_Bool color;
int channels;
uint16_t startx = 0, endx, pixels;
int move = 0;
DBG (DBG_proc, "setup_for_scan: start\n");
DBG (DBG_info,
"setup_for_scan settings:\nResolution: %ux%uDPI\n"
"Lines : %u\nPixels : %u\nStartpos : %.3f/%.3f\nScan mode : %d\nScan method: %s\n\n",
settings.xres, settings.yres, settings.lines, settings.pixels,
settings.tl_x, settings.tl_y, settings.scan_mode,
settings.scan_method == SCAN_METHOD_FLATBED ? "flatbed" : "XPA");
if (settings.scan_mode == SCAN_MODE_COLOR) /* single pass color */
{
channels = 3;
color = SANE_TRUE;
}
else
{
channels = 1;
color = SANE_FALSE;
}
/* compute distance to move */
move = 0;
/* XXX STEF XXX MD5345 -> optical_ydpi, other base_ydpi => half/full step ? */
if (split == SANE_FALSE)
{
if (dev->model->is_sheetfed == SANE_FALSE)
{
if (ycorrection == SANE_TRUE)
{
move =
(SANE_UNFIX (dev->model->y_offset) *
dev->motor.optical_ydpi) / MM_PER_INCH;
}
/* add tl_y to base movement */
move += (settings.tl_y * dev->motor.optical_ydpi) / MM_PER_INCH;
}
else
{
move += (settings.tl_y * dev->motor.optical_ydpi) / MM_PER_INCH;
}
DBG (DBG_info, "setup_for_scan: move=%d steps\n", move);
/* security check */
if (move < 0)
{
DBG (DBG_error,
"setup_for_scan: overriding negative move value %d\n", move);
move = 0;
}
}
DBG (DBG_info, "setup_for_scan: move=%d steps\n", move);
/* pixels are allways given at full CCD optical resolution */
/* use detected left margin and fixed value */
if (xcorrection == SANE_TRUE)
{
if (dev->sensor.CCD_start_xoffset > 0)
startx = dev->sensor.CCD_start_xoffset;
else
startx = dev->sensor.dummy_pixel;
if (settings.scan_method == SCAN_METHOD_FLATBED)
{
startx +=
((SANE_UNFIX (dev->model->x_offset) * dev->sensor.optical_res) /
MM_PER_INCH);
}
else
{
startx +=
((SANE_UNFIX (dev->model->x_offset_ta) *
dev->sensor.optical_res) / MM_PER_INCH);
}
}
else
{
/* startx cannot be below dummy pixel value */
startx = dev->sensor.dummy_pixel;
}
/* add x coordinates : expressed in sensor max dpi */
startx += (settings.tl_x * dev->sensor.optical_res) / MM_PER_INCH;
/* stagger works with odd start cordinates */
if (dev->model->flags & GENESYS_FLAG_STAGGERED_LINE)
startx |= 1;
pixels = (settings.pixels * dev->sensor.optical_res) / settings.xres;
/* special requirement for 400 dpi on 1200 dpi sensors */
if (settings.xres == 400)
{
pixels = (pixels / 6) * 6;
}
endx = startx + pixels;
/* TODO check for pixel width overflow */
/* set up correct values for scan (gamma and shading enabled) */
status = gl646_setup_registers (dev,
dev->reg,
settings,
dev->slope_table0,
dev->slope_table1,
settings.xres,
move,
settings.lines,
startx, endx, color, settings.depth);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"setup_for_scan: failed setup registers: %s\n",
sane_strstatus (status));
return status;
}
/* now post-process values for register and options fine tuning */
/* select color filter based on settings */
dev->reg[reg_0x04].value &= ~REG04_FILTER;
if (channels == 1)
{
switch (settings.color_filter)
{
/* red */
case 0:
dev->reg[reg_0x04].value |= 0x04;
break;
/* green */
case 1:
dev->reg[reg_0x04].value |= 0x08;
break;
/* blue */
case 2:
dev->reg[reg_0x04].value |= 0x0c;
break;
default:
break;
}
}
/* send computed slope tables */
status =
gl646_send_slope_table (dev, 0, dev->slope_table0,
sanei_genesys_read_reg_from_set (dev->reg, 0x21));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"setup_for_scan: failed to send slope table 0: %s\n",
sane_strstatus (status));
return status;
}
status =
gl646_send_slope_table (dev, 1, dev->slope_table1,
sanei_genesys_read_reg_from_set (dev->reg, 0x6b));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"setup_for_scan: failed to send slope table 1: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_proc, "setup_for_scan: end\n");
return status;
}
/*
* this function sends generic gamma table (ie linear ones)
* or the Sensor specific one if provided
*/
static SANE_Status
gl646_send_gamma_table (Genesys_Device * dev, SANE_Bool generic)
{
int size;
int address;
int status;
uint8_t *gamma;
int i;
/* don't send anything if no specific gamma table defined */
if (!generic
&& (dev->sensor.red_gamma_table == NULL
|| dev->sensor.green_gamma_table == NULL
|| dev->sensor.blue_gamma_table == NULL))
{
DBG (DBG_proc, "gl646_send_gamma_table: nothing to send, skipping\n");
return SANE_STATUS_GOOD;
}
/* gamma table size */
if (dev->reg[reg_0x05].value & REG05_GMMTYPE)
size = 16384;
else
size = 4096;
/* allocate temporary gamma tables: 16 bits words, 3 channels */
gamma = (uint8_t *) malloc (size * 2 * 3);
if (!gamma)
return SANE_STATUS_NO_MEM;
/* take care off generic/specific data */
if (generic)
{
/* fill with default values */
for (i = 0; i < size; i++)
{
gamma[i * 2] = i & 0xff;
gamma[i * 2 + 1] = i >> 8;
gamma[i * 2 + size * 2] = i & 0xff;
gamma[i * 2 + 1 + size * 2] = i >> 8;
gamma[i * 2 + size * 4] = i & 0xff;
gamma[i * 2 + 1 + size * 4] = i >> 8;
}
}
else
{
/* copy sensor specific's gamma tables */
for (i = 0; i < size; i++)
{
gamma[i * 2] = dev->sensor.red_gamma_table[i] & 0xff;
gamma[i * 2 + 1] = dev->sensor.red_gamma_table[i] >> 8;
gamma[i * 2 + size * 2] = dev->sensor.green_gamma_table[i] & 0xff;
gamma[i * 2 + 1 + size * 2] = dev->sensor.green_gamma_table[i] >> 8;
gamma[i * 2 + size * 4] = dev->sensor.blue_gamma_table[i] & 0xff;
gamma[i * 2 + 1 + size * 4] = dev->sensor.blue_gamma_table[i] >> 8;
}
}
/* table address */
switch (dev->reg[reg_0x05].value >> 6)
{
case 0: /* 600 dpi */
address = 0x09000;
break;
case 1: /* 1200 dpi */
address = 0x11000;
break;
case 2: /* 2400 dpi */
address = 0x20000;
break;
default:
free (gamma);
return SANE_STATUS_INVAL;
}
/* send address */
status = sanei_genesys_set_buffer_address (dev, address);
if (status != SANE_STATUS_GOOD)
{
free (gamma);
DBG (DBG_error,
"gl646_send_gamma_table: failed to set buffer address: %s\n",
sane_strstatus (status));
return status;
}
/* send data */
status = gl646_bulk_write_data (dev, 0x3c, (uint8_t *) gamma, size * 2 * 3);
if (status != SANE_STATUS_GOOD)
{
free (gamma);
DBG (DBG_error,
"gl646_send_gamma_table: failed to send gamma table: %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_proc, "gl646_send_gamma_table: end\n");
free (gamma);
return SANE_STATUS_GOOD;
}
/** @brief this function does the led calibration.
* this function does the led calibration by scanning one line of the calibration
* area below scanner's top on white strip. The scope of this function is
* currently limited to the XP200
*/
static SANE_Status
gl646_led_calibration (Genesys_Device * dev)
{
int total_size;
uint8_t *line;
unsigned int i, j;
SANE_Status status = SANE_STATUS_GOOD;
int val;
unsigned int channels;
int avg[3], avga, avge;
int turn;
char fn[20];
uint16_t expr, expg, expb;
Genesys_Settings settings;
SANE_Int resolution;
SANE_Bool acceptable = SANE_FALSE;
DBG (DBG_proc, "gl646_led_calibration\n");
if (!dev->model->is_cis)
{
DBG (DBG_proc,
"gl646_led_calibration: not a cis scanner, nothing to do...\n");
return SANE_STATUS_GOOD;
}
/* get led calibration resolution */
if (dev->settings.scan_mode == SCAN_MODE_COLOR)
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->sensor.optical_res,
SANE_TRUE);
settings.scan_mode = SCAN_MODE_COLOR;
channels = 3;
}
else
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->sensor.optical_res,
SANE_FALSE);
settings.scan_mode = SCAN_MODE_GRAY;
channels = 1;
}
/* offset calibration is always done in color mode */
settings.scan_method = SCAN_METHOD_FLATBED;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
settings.lines = 1;
settings.depth = 16;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* colors * bytes_per_color * scan lines */
total_size = settings.pixels * channels * 2 * 1;
line = malloc (total_size);
if (!line)
{
DBG (DBG_error, "gl646_led_calibration: Failed to allocate %d bytes\n",
total_size);
return SANE_STATUS_NO_MEM;
}
/*
we try to get equal bright leds here:
loop:
average per color
adjust exposure times
Sensor_Master uint8_t regs_0x10_0x15[6];
*/
expr = (dev->sensor.regs_0x10_0x1d[0] << 8) | dev->sensor.regs_0x10_0x1d[1];
expg = (dev->sensor.regs_0x10_0x1d[2] << 8) | dev->sensor.regs_0x10_0x1d[3];
expb = (dev->sensor.regs_0x10_0x1d[4] << 8) | dev->sensor.regs_0x10_0x1d[5];
turn = 0;
do
{
dev->sensor.regs_0x10_0x1d[0] = (expr >> 8) & 0xff;
dev->sensor.regs_0x10_0x1d[1] = expr & 0xff;
dev->sensor.regs_0x10_0x1d[2] = (expg >> 8) & 0xff;
dev->sensor.regs_0x10_0x1d[3] = expg & 0xff;
dev->sensor.regs_0x10_0x1d[4] = (expb >> 8) & 0xff;
dev->sensor.regs_0x10_0x1d[5] = expb & 0xff;
DBG (DBG_info, "gl646_led_calibration: starting first line reading\n");
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE, &line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_led_calibration: Failed to setup scan: %s\n",
sane_strstatus (status));
return status;
}
if (DBG_LEVEL >= DBG_data)
{
snprintf (fn, 20, "led_%02d.pnm", turn);
sanei_genesys_write_pnm_file (fn,
line,
16, channels, settings.pixels, 1);
}
acceptable = SANE_TRUE;
for (j = 0; j < channels; j++)
{
avg[j] = 0;
for (i = 0; i < settings.pixels; i++)
{
if (dev->model->is_cis)
val =
line[i * 2 + j * 2 * settings.pixels + 1] * 256 +
line[i * 2 + j * 2 * settings.pixels];
else
val =
line[i * 2 * channels + 2 * j + 1] * 256 +
line[i * 2 * channels + 2 * j];
avg[j] += val;
}
avg[j] /= settings.pixels;
}
DBG (DBG_info, "gl646_led_calibration: average: "
"%d,%d,%d\n", avg[0], avg[1], avg[2]);
acceptable = SANE_TRUE;
if (!acceptable)
{
avga = (avg[0] + avg[1] + avg[2]) / 3;
expr = (expr * avga) / avg[0];
expg = (expg * avga) / avg[1];
expb = (expb * avga) / avg[2];
/* keep exposure time in a working window */
avge = (expr + expg + expb) / 3;
if (avge > 0x2000)
{
expr = (expr * 0x2000) / avge;
expg = (expg * 0x2000) / avge;
expb = (expb * 0x2000) / avge;
}
if (avge < 0x400)
{
expr = (expr * 0x400) / avge;
expg = (expg * 0x400) / avge;
expb = (expb * 0x400) / avge;
}
}
turn++;
}
while (!acceptable && turn < 100);
DBG (DBG_info,
"gl646_led_calibration: acceptable exposure: 0x%04x,0x%04x,0x%04x\n",
expr, expg, expb);
/* cleanup before return */
free (line);
DBG (DBG_proc, "gl646_led_calibration: completed\n");
return status;
}
/**
* average dark pixels of a scan
*/
static int
dark_average (uint8_t * data, unsigned int pixels, unsigned int lines,
unsigned int channels, unsigned int black)
{
unsigned int i, j, k, average, count;
unsigned int avg[3];
uint8_t val;
/* computes average value on black margin */
for (k = 0; k < channels; k++)
{
avg[k] = 0;
count = 0;
for (i = 0; i < lines; i++)
{
for (j = 0; j < black; j++)
{
val = data[i * channels * pixels + j + k];
avg[k] += val;
count++;
}
}
if (count)
avg[k] /= count;
DBG (DBG_info, "dark_average: avg[%d] = %d\n", k, avg[k]);
}
average = 0;
for (i = 0; i < channels; i++)
average += avg[i];
average /= channels;
DBG (DBG_info, "dark_average: average = %d\n", average);
return average;
}
/** @brief calibration for AD frontend devices
* we do simple scan until all black_pixels are higher than 0,
* raising offset at each turn.
*/
static SANE_Status
ad_fe_offset_calibration (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
uint8_t *line;
unsigned int channels;
char title[32];
int pass = 0;
SANE_Int resolution;
Genesys_Settings settings;
unsigned int x, y, adr, min;
unsigned int bottom, black_pixels;
DBG (DBG_proc, "ad_fe_offset_calibration: start\n");
resolution =
get_closest_resolution (dev->model->ccd_type, dev->sensor.optical_res,
SANE_TRUE);
channels = 3;
black_pixels =
(dev->sensor.black_pixels * resolution) / dev->sensor.optical_res;
DBG (DBG_io2, "ad_fe_offset_calibration: black_pixels=%d\n", black_pixels);
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
settings.lines = CALIBRATION_LINES;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* scan first line of data with no gain */
dev->frontend.gain[0] = 0;
dev->frontend.gain[1] = 0;
dev->frontend.gain[2] = 0;
/* scan with no move */
bottom = 1;
do
{
pass++;
dev->frontend.offset[0] = bottom;
dev->frontend.offset[1] = bottom;
dev->frontend.offset[2] = bottom;
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE, &line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"ad_fe_offset_calibration: failed to scan first line\n");
return status;
}
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "offset%03d.pnm", bottom);
sanei_genesys_write_pnm_file (title, line, 8, channels,
settings.pixels, settings.lines);
}
min = 0;
for (y = 0; y < settings.lines; y++)
{
for (x = 0; x < black_pixels; x++)
{
adr = (x + y * settings.pixels) * channels;
if (line[adr] > min)
min = line[adr];
if (line[adr + 1] > min)
min = line[adr + 1];
if (line[adr + 2] > min)
min = line[adr + 2];
}
}
free (line);
DBG (DBG_io2, "ad_fe_offset_calibration: pass=%d, min=%d\n", pass, min);
bottom++;
}
while (pass < 128 && min == 0);
if (pass == 128)
{
DBG (DBG_error,
"ad_fe_offset_calibration: failed to find correct offset\n");
return SANE_STATUS_INVAL;
}
DBG (DBG_info, "ad_fe_offset_calibration: offset=(%d,%d,%d)\n",
dev->frontend.offset[0], dev->frontend.offset[1],
dev->frontend.offset[2]);
DBG (DBG_proc, "ad_fe_offset_calibration: end\n");
return status;
}
#define DARK_TARGET 8
/**
* 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.
* genesys_search_start() must have been called so that the offsets and margins
* are already known.
* @param dev scanner's device
* @return SANE_STATUS_GOOD if success, else error code is failure
*/
static SANE_Status
gl646_offset_calibration (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
uint8_t *first_line, *second_line;
unsigned int channels;
char title[32];
int pass = 0, avg;
SANE_Int resolution;
Genesys_Settings settings;
int topavg, bottomavg;
int top, bottom, black_pixels;
/* Analog Device fronted have a different calibration */
if (dev->model->dac_type == DAC_AD_XP200)
{
return ad_fe_offset_calibration (dev);
}
DBG (DBG_proc, "gl646_offset_calibration: start\n");
/* setup for a RGB scan, one full sensor's width line */
/* resolution is the one from the final scan */
if (dev->settings.xres > dev->sensor.optical_res)
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->sensor.optical_res,
SANE_TRUE);
}
else
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->settings.xres,
SANE_TRUE);
}
channels = 3;
black_pixels =
(dev->sensor.black_pixels * resolution) / dev->sensor.optical_res;
DBG (DBG_io2, "gl646_offset_calibration: black_pixels=%d\n", black_pixels);
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
settings.lines = CALIBRATION_LINES;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* scan first line of data with no gain, but with offset from
* last calibration */
dev->frontend.gain[0] = 0;
dev->frontend.gain[1] = 0;
dev->frontend.gain[2] = 0;
/* scan with no move */
bottom = 90;
dev->frontend.offset[0] = bottom;
dev->frontend.offset[1] = bottom;
dev->frontend.offset[2] = bottom;
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE,
&first_line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_offset_calibration: failed to scan first line\n");
return status;
}
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "offset%03d.pnm", bottom);
sanei_genesys_write_pnm_file (title, first_line, 8, channels,
settings.pixels, settings.lines);
}
bottomavg =
dark_average (first_line, settings.pixels, settings.lines, channels,
black_pixels);
free (first_line);
DBG (DBG_io2, "gl646_offset_calibration: bottom avg=%d\n", bottomavg);
/* now top value */
top = 231;
dev->frontend.offset[0] = top;
dev->frontend.offset[1] = top;
dev->frontend.offset[2] = top;
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE,
&second_line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_offset_calibration: failed to scan first line\n");
return status;
}
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "offset%03d.pnm", top);
sanei_genesys_write_pnm_file (title, second_line, 8, channels,
settings.pixels, settings.lines);
}
topavg =
dark_average (second_line, settings.pixels, settings.lines, channels,
black_pixels);
free (second_line);
DBG (DBG_io2, "gl646_offset_calibration: top avg=%d\n", topavg);
/* loop until acceptable level */
while ((pass < 32) && (top - bottom > 1))
{
pass++;
/* settings for new scan */
dev->frontend.offset[0] = (top + bottom) / 2;
dev->frontend.offset[1] = (top + bottom) / 2;
dev->frontend.offset[2] = (top + bottom) / 2;
/* scan with no move */
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE,
&second_line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_offset_calibration: failed to scan first line\n");
return status;
}
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "offset%03d.pnm", dev->frontend.offset[1]);
sanei_genesys_write_pnm_file (title, second_line, 8, channels,
settings.pixels, settings.lines);
}
avg =
dark_average (second_line, settings.pixels, settings.lines, channels,
black_pixels);
DBG (DBG_info, "gl646_offset_calibration: avg=%d offset=%d\n", avg,
dev->frontend.offset[1]);
free (second_line);
/* compute new boundaries */
if (topavg == avg)
{
topavg = avg;
top = dev->frontend.offset[1];
}
else
{
bottomavg = avg;
bottom = dev->frontend.offset[1];
}
}
/* in case of debug do a final scan to get result */
if (DBG_LEVEL >= DBG_data)
{
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE,
&second_line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_offset_calibration: failed to scan final line\n");
return status;
}
sanei_genesys_write_pnm_file ("offset-final.pnm", second_line, 8,
channels, settings.pixels,
settings.lines);
free (second_line);
}
DBG (DBG_info, "gl646_offset_calibration: offset=(%d,%d,%d)\n",
dev->frontend.offset[0], dev->frontend.offset[1],
dev->frontend.offset[2]);
DBG (DBG_proc, "gl646_offset_calibration: end\n");
return status;
}
/** @brief gain calibration for Analog Device frontends
* Alternative coarse gain calibration
*/
static SANE_Status
ad_fe_coarse_gain_calibration (Genesys_Device * dev, int dpi)
{
uint8_t *line;
unsigned int i, channels, val;
unsigned int size, count, resolution, pass;
SANE_Status status = SANE_STATUS_GOOD;
float average;
Genesys_Settings settings;
char title[32];
DBG (DBG_proc, "ad_fe_coarse_gain_calibration: start\n");
/* setup for a RGB scan, one full sensor's width line */
/* resolution is the one from the final scan */
resolution = get_closest_resolution (dev->model->ccd_type, dpi, SANE_TRUE);
channels = 3;
settings.scan_mode = SCAN_MODE_COLOR;
settings.scan_method = SCAN_METHOD_FLATBED;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
settings.lines = CALIBRATION_LINES;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
size = channels * settings.pixels * settings.lines;
/* start gain value */
dev->frontend.gain[0] = 1;
dev->frontend.gain[1] = 1;
dev->frontend.gain[2] = 1;
average = 0;
pass = 0;
/* loop until each channel raises to acceptable level */
while ((average < dev->sensor.gain_white_ref) && (pass < 30))
{
/* scan with no move */
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE, &line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"ad_fe_coarse_gain_calibration: failed to scan first line\n");
return status;
}
/* log scanning data */
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "alternative_coarse%02d.pnm", pass);
sanei_genesys_write_pnm_file (title, line, 8,
channels, settings.pixels,
settings.lines);
}
pass++;
/* computes white average */
average = 0;
count = 0;
for (i = 0; i < size; i++)
{
val = line[i];
average += val;
count++;
}
average = average / count;
/* adjusts gain for the channel */
if (average < dev->sensor.gain_white_ref)
dev->frontend.gain[0]++;
dev->frontend.gain[1] = dev->frontend.gain[0];
dev->frontend.gain[2] = dev->frontend.gain[0];
DBG (DBG_proc,
"ad_fe_coarse_gain_calibration: average = %.2f, gain = %d\n",
average, dev->frontend.gain[0]);
free (line);
}
DBG (DBG_info, "ad_fe_coarse_gain_calibration: gains=(%d,%d,%d)\n",
dev->frontend.gain[0], dev->frontend.gain[1], dev->frontend.gain[2]);
DBG (DBG_proc, "ad_fe_coarse_gain_calibration: end\n");
return status;
}
/**
* Alternative coarse gain calibration
* this on uses the settings from offset_calibration. First scan moves so
* we can go to calibration area for XPA.
* @param dev device for scan
* @param dpi resolutnio to calibrate at
*/
static SANE_Status
gl646_coarse_gain_calibration (Genesys_Device * dev, int dpi)
{
uint8_t *line;
unsigned int i, j, k, channels, val, maximum, idx;
unsigned int size, count, resolution, pass;
SANE_Status status = SANE_STATUS_GOOD;
float average[3];
Genesys_Settings settings;
char title[32];
if (dev->model->ccd_type == CIS_XP200)
{
return ad_fe_coarse_gain_calibration (dev, dev->sensor.optical_res);
}
DBG (DBG_proc, "gl646_coarse_gain_calibration: start\n");
/* setup for a RGB scan, one full sensor's width line */
/* resolution is the one from the final scan */
channels = 3;
/* we are searching a sensor resolution */
if (dpi > dev->sensor.optical_res)
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->sensor.optical_res,
SANE_TRUE);
}
else
{
resolution =
get_closest_resolution (dev->model->ccd_type, dev->settings.xres,
SANE_TRUE);
}
settings.scan_method = dev->settings.scan_method;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_y = 0;
if (settings.scan_method == SCAN_METHOD_FLATBED)
{
settings.tl_x = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
}
else
{
settings.tl_x = SANE_UNFIX (dev->model->x_offset_ta);
settings.pixels =
(SANE_UNFIX (dev->model->x_size_ta) * resolution) / MM_PER_INCH;
}
settings.lines = CALIBRATION_LINES;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
size = channels * settings.pixels * settings.lines;
/* start gain value */
dev->frontend.gain[0] = 1;
dev->frontend.gain[1] = 1;
dev->frontend.gain[2] = 1;
if (channels > 1)
{
average[0] = 0;
average[1] = 0;
average[2] = 0;
idx = 0;
}
else
{
average[0] = 255;
average[1] = 255;
average[2] = 255;
idx = dev->settings.color_filter;
average[idx] = 0;
}
pass = 0;
/* loop until each channel raises to acceptable level */
while (((average[0] < dev->sensor.gain_white_ref)
|| (average[1] < dev->sensor.gain_white_ref)
|| (average[2] < dev->sensor.gain_white_ref)) && (pass < 30))
{
/* scan with no move */
status =
simple_scan (dev, settings, SANE_FALSE, SANE_TRUE, SANE_FALSE, &line);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_coarse_gain_calibration: failed to scan first line\n");
return status;
}
/* log scanning data */
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "coarse_gain%02d.pnm", pass);
sanei_genesys_write_pnm_file (title, line, 8,
channels, settings.pixels,
settings.lines);
}
pass++;
/* 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 (k = idx; k < idx + channels; k++)
{
/* we find the maximum white value, so we can deduce a threshold
to average white values */
maximum = 0;
for (i = 0; i < settings.lines; i++)
{
for (j = 0; j < settings.pixels; j++)
{
val = line[i * channels * settings.pixels + j + k];
if (val > maximum)
maximum = val;
}
}
/* threshold */
maximum *= 0.9;
/* computes white average */
average[k] = 0;
count = 0;
for (i = 0; i < settings.lines; i++)
{
for (j = 0; j < settings.pixels; j++)
{
/* averaging only white points allow us not to care about dark margins */
val = line[i * channels * settings.pixels + j + k];
if (val > maximum)
{
average[k] += val;
count++;
}
}
}
average[k] = average[k] / count;
/* adjusts gain for the channel */
if (average[k] < dev->sensor.gain_white_ref)
dev->frontend.gain[k]++;
DBG (DBG_proc,
"gl646_coarse_gain_calibration: channel %d, average = %.2f, gain = %d\n",
k, average[k], dev->frontend.gain[k]);
}
free (line);
}
if (channels < 3)
{
dev->frontend.gain[1] = dev->frontend.gain[0];
dev->frontend.gain[2] = dev->frontend.gain[0];
}
DBG (DBG_info, "gl646_coarse_gain_calibration: gains=(%d,%d,%d)\n",
dev->frontend.gain[0], dev->frontend.gain[1], dev->frontend.gain[2]);
DBG (DBG_proc, "gl646_coarse_gain_calibration: end\n");
return status;
}
/**
* sets up the scanner's register for warming up. We scan 2 lines without moving.
*
*/
static SANE_Status
gl646_init_regs_for_warmup (Genesys_Device * dev,
Genesys_Register_Set * local_reg,
int *channels, int *total_size)
{
SANE_Status status = SANE_STATUS_GOOD;
Genesys_Settings settings;
int resolution, lines;
DBG (DBG_proc, "gl646_init_regs_for_warmup: start\n");
sanei_genesys_init_fe (dev);
resolution = get_closest_resolution (dev->model->ccd_type, 300, SANE_FALSE);
/* set up for a half width 2 lines color scan without moving */
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_GRAY;
settings.xres = resolution;
settings.yres = resolution;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels =
(dev->sensor.sensor_pixels * resolution) / dev->sensor.optical_res;
settings.lines = 2;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* setup for scan */
status = setup_for_scan (dev, settings, SANE_TRUE, SANE_FALSE, SANE_FALSE);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_init_regs_for_warmup: setup_for_scan failed (%s)\n",
sane_strstatus (status));
return status;
}
/* we are not going to move, so clear these bits */
dev->reg[reg_0x02].value &= ~(REG02_FASTFED | REG02_AGOHOME);
/* don't enable any correction for this scan */
dev->reg[reg_0x01].value &= ~REG01_DVDSET;
/* XXX STEF XXX
dev->reg[reg_0x05].value &= ~REG05_GMMENB; */
/* copy to local_reg */
memcpy (local_reg, dev->reg, (GENESYS_GL646_MAX_REGS + 1) * sizeof (Genesys_Register_Set));
/* turn off motor during this scan */
gl646_set_motor_power (local_reg, SANE_FALSE);
/* returned value to higher level warmup function */
*channels = 1;
lines = gl646_get_triple_reg (local_reg, REG_LINCNT) + 1;
*total_size = lines * settings.pixels;
/* now registers are ok, write them to scanner */
RIE (gl646_set_fe (dev, AFE_SET, settings.xres));
RIE (gl646_bulk_write_register (dev, local_reg, GENESYS_GL646_MAX_REGS));
DBG (DBG_proc, "gl646_init_regs_for_warmup: end\n");
return status;
}
/*
* this function moves head without scanning, forward, then backward
* so that the head goes to park position.
* as a by-product, also check for lock
*/
static SANE_Status
gl646_repark_head (Genesys_Device * dev)
{
SANE_Status status;
Genesys_Settings settings;
unsigned int expected, steps;
DBG (DBG_proc, "gl646_repark_head: start\n");
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres =
get_closest_resolution (dev->model->ccd_type, 75, SANE_FALSE);
settings.yres = settings.xres;
settings.tl_x = 0;
settings.tl_y = 5;
settings.pixels = 600;
settings.lines = 4;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
status = setup_for_scan (dev, settings, SANE_FALSE, SANE_FALSE, SANE_FALSE);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_repark_head: failed to setup for scan: %s\n",
sane_strstatus (status));
return status;
}
/* TODO seems wrong ... no effective scan */
dev->reg[reg_0x01].value &= ~REG01_SCAN;
status = gl646_bulk_write_register (dev, dev->reg, GENESYS_GL646_MAX_REGS);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_repark_head: failed to send registers: %s\n",
sane_strstatus (status));
return status;
}
/* start scan */
status = gl646_begin_scan (dev, dev->reg, SANE_TRUE);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_repark_head: failed to begin scan: \n");
return status;
}
expected = gl646_get_triple_reg (dev->reg, REG_FEEDL);
do
{
usleep (100 * 1000);
status = sanei_genesys_read_feed_steps (dev, &steps);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_repark_head: failed to read feed steps: %s\n",
sane_strstatus (status));
return status;
}
}
while (steps < expected);
/* toggle motor flag, put an huge step number and redo move backward */
status = gl646_slow_back_home (dev, 1);
DBG (DBG_proc, "gl646_repark_head: end\n");
return status;
}
/* *
* initialize ASIC : registers, motor tables, and gamma tables
* then ensure scanner's head is at home
* @param dev device description of the scanner to initailize
* @return SANE_STATUS_GOOD if success, error code if failure
*/
static SANE_Status
gl646_init (Genesys_Device * dev)
{
SANE_Status status;
struct timeval tv;
uint8_t cold = 0, val = 0;
uint32_t addr = 0xdead;
int size;
size_t len;
DBG_INIT ();
DBG (DBG_proc, "gl646_init: start\n");
/* to detect real power up condition, we write to REG41
* with pwrbit set, then read it back. When scanner is cold (just replugged)
* PWRBIT will be set in the returned value
*/
RIE (sanei_genesys_get_status (dev, &cold));
DBG (DBG_info, "gl646_init: status=0x%02x\n", cold);
cold = !(cold & REG41_PWRBIT);
if (cold)
{
DBG (DBG_info, "gl646_init: device is cold\n");
}
else
{
DBG (DBG_info, "gl646_init: device is hot\n");
}
/* if scanning session hasn't been initialized, set it up */
if (!dev->already_initialized)
{
dev->dark_average_data = NULL;
dev->white_average_data = NULL;
dev->settings.color_filter = 1; /* green filter by default */
gettimeofday (&tv, NULL);
dev->init_date = tv.tv_sec;
switch (dev->model->motor_type)
{
/* set to 11111 to spot bugs, sanei_genesys_exposure_time should
have obsoleted this field */
case MOTOR_5345:
dev->settings.exposure_time = 11111;
break;
case MOTOR_ST24:
dev->settings.exposure_time = 11000;
break;
default:
dev->settings.exposure_time = 11000;
break;
}
/* Set default values for registers */
gl646_init_regs (dev);
/* build default gamma tables */
if (dev->reg[reg_0x05].value & REG05_GMMTYPE)
size = 16384;
else
size = 4096;
if (dev->sensor.red_gamma_table == NULL)
{
dev->sensor.red_gamma_table = (uint16_t *) malloc (2 * size);
if (dev->sensor.red_gamma_table == NULL)
{
DBG (DBG_error,
"gl646_init: could not allocate memory for gamma table\n");
return SANE_STATUS_NO_MEM;
}
sanei_genesys_create_gamma_table (dev->sensor.red_gamma_table,
size, size - 1, size - 1,
dev->sensor.red_gamma);
}
if (dev->sensor.green_gamma_table == NULL)
{
dev->sensor.green_gamma_table = (uint16_t *) malloc (2 * size);
if (dev->sensor.red_gamma_table == NULL)
{
DBG (DBG_error,
"gl646_init: could not allocate memory for gamma table\n");
return SANE_STATUS_NO_MEM;
}
sanei_genesys_create_gamma_table (dev->sensor.green_gamma_table,
size, size - 1, size - 1,
dev->sensor.green_gamma);
}
if (dev->sensor.blue_gamma_table == NULL)
{
dev->sensor.blue_gamma_table = (uint16_t *) malloc (2 * size);
if (dev->sensor.red_gamma_table == NULL)
{
DBG (DBG_error,
"gl646_init: could not allocate memory for gamma table\n");
return SANE_STATUS_NO_MEM;
}
sanei_genesys_create_gamma_table (dev->sensor.blue_gamma_table,
size, size - 1, size - 1,
dev->sensor.blue_gamma);
}
/* Init shading data */
RIE (sanei_genesys_init_shading_data (dev, dev->sensor.sensor_pixels));
/* initial calibration reg values */
memcpy (dev->calib_reg, dev->reg,
(GENESYS_GL646_MAX_REGS + 1) * sizeof (Genesys_Register_Set));
}
/* execute physical unit init only if cold */
if (cold)
{
DBG (DBG_info, "gl646_init: device is cold\n");
val = 0x04;
RIE (sanei_usb_control_msg
(dev->dn, REQUEST_TYPE_OUT, REQUEST_REGISTER, VALUE_INIT,
INDEX, 1, &val));
/* ASIC reset */
RIE (sanei_genesys_write_register (dev, 0x0e, 0x00));
usleep (100000UL); /* sleep 100 ms */
/* Write initial registers */
RIE (gl646_bulk_write_register (dev, dev->reg, GENESYS_GL646_MAX_REGS));
/* Test ASIC and RAM */
if (!(dev->model->flags & GENESYS_FLAG_LAZY_INIT))
{
RIE (gl646_asic_test (dev));
}
/* send gamma tables if needed */
status = gl646_send_gamma_table (dev, 1);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_init: failed to send generic gamma tables: %s\n",
sane_strstatus (status));
return status;
}
/* Set powersaving (default = 15 minutes) */
RIE (gl646_set_powersaving (dev, 15));
} /* end if cold */
/* Set analog frontend */
RIE (gl646_set_fe (dev, AFE_INIT, 0));
/* GPO enabling for XP200 */
if (dev->model->ccd_type == CIS_XP200)
{
sanei_genesys_write_register (dev, 0x68, dev->gpo.enable[0]);
sanei_genesys_write_register (dev, 0x69, dev->gpo.enable[1]);
/* enable GPIO */
val = 6;
status = gl646_gpio_output_enable (dev->dn, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_init: GPO enable failed ... %s\n",
sane_strstatus (status));
}
val = 0;
/* writes 0 to GPIO */
status = gl646_gpio_write (dev->dn, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_init: GPO write failed ... %s\n",
sane_strstatus (status));
}
/* clear GPIO enable */
status = gl646_gpio_output_enable (dev->dn, val);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_init: GPO disable failed ... %s\n",
sane_strstatus (status));
}
sanei_genesys_write_register (dev, 0x66, 0x10);
sanei_genesys_write_register (dev, 0x66, 0x00);
sanei_genesys_write_register (dev, 0x66, 0x10);
}
/* MD6471/G2410 and XP200 read/write data from an undocumented memory area which
* is after the second slope table */
if (dev->model->gpo_type != GPO_HP3670
&& dev->model->gpo_type != GPO_HP2400)
{
switch (dev->sensor.optical_res)
{
case 600:
addr = 0x08200;
break;
case 1200:
addr = 0x10200;
break;
case 2400:
addr = 0x1fa00;
break;
}
status = sanei_genesys_set_buffer_address (dev, addr);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_init: failed to set up control address\n");
return SANE_STATUS_INVAL;
}
sanei_usb_set_timeout (2 * 1000);
len = 6;
status = gl646_bulk_read_data (dev, 0x45, dev->control, len);
/* for some reason, read fails here for MD6471, HP2300 and XP200
* one time out of 2 scanimage launches
*/
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_warn, "gl646_init: failed to read control\n");
status = gl646_bulk_read_data (dev, 0x45, dev->control, len);
}
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_warn, "gl646_init: failed to read control\n");
return SANE_STATUS_INVAL;
}
else
{
DBG (DBG_info,
"gl646_init: control read=0x%02x 0x%02x 0x%02x 0x%02x 0x%02x 0x%02x\n",
dev->control[0], dev->control[1], dev->control[2],
dev->control[3], dev->control[4], dev->control[5]);
}
sanei_usb_set_timeout (30 * 1000);
}
else
/* HP2400 and HP3670 case */
{
dev->control[0] = 0x00;
dev->control[1] = 0x00;
dev->control[2] = 0x01;
dev->control[3] = 0x00;
dev->control[4] = 0x00;
dev->control[5] = 0x00;
}
/* ensure head is correctly parked, and check lock */
if (dev->model->is_sheetfed == SANE_FALSE)
{
if (dev->model->flags & GENESYS_FLAG_REPARK)
{
status = gl646_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");
#ifdef SANE_STATUS_HW_LOCKED
return SANE_STATUS_HW_LOCKED;
#else
return SANE_STATUS_JAMMED;
#endif
}
else
DBG (DBG_error,
"gl646_init: gl646_repark_head failed: %s\n",
sane_strstatus (status));
return status;
}
}
else
{
RIE (gl646_slow_back_home (dev, SANE_TRUE));
}
}
/* here session and device are initialized */
dev->already_initialized = SANE_TRUE;
DBG (DBG_proc, "gl646_init: end\n");
return SANE_STATUS_GOOD;
}
#ifndef UNIT_TESTING
static
#endif
SANE_Status
gl646_move_to_ta (Genesys_Device * dev)
{
SANE_Status status = SANE_STATUS_GOOD;
DBG (DBG_proc, "gl646_move_to_ta: starting\n");
if (simple_move (dev, SANE_UNFIX (dev->model->y_offset_calib_ta)) !=
SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_move_to_ta: failed to move to calibration area\n");
return status;
}
DBG (DBG_proc, "gl646_move_to_ta: end\n");
return status;
}
/**
* Does a simple scan: ie no line reordering and avanced data buffering and
* shading correction. Memory for data is allocated in this function
* and must be freed by caller.
* @param dev device of the scanner
* @param settings parameters of the scan
* @param move SANE_TRUE if moving during scan
* @param move SANE_TRUE if moving forward during scan
* @param data pointer for the data
*/
#ifndef UNIT_TESTING
static
#endif
SANE_Status
simple_scan (Genesys_Device * dev, Genesys_Settings settings, SANE_Bool move,
SANE_Bool forward, SANE_Bool shading, unsigned char **data)
{
SANE_Status status = SANE_STATUS_INVAL;
unsigned int size, lines, x, y, bpp;
SANE_Bool empty, split;
unsigned char *buffer;
int count;
uint8_t val;
DBG (DBG_proc, "simple_scan: starting\n");
DBG (DBG_io, "simple_scan: move=%d, forward=%d, shading=%d\n", move,
forward, shading);
/* round up to multiple of 3 in case of CIS scanner */
if (dev->model->is_cis == SANE_TRUE)
{
settings.lines = ((settings.lines + 2) / 3) * 3;
}
/* setup for move then scan */
if (move == SANE_TRUE && settings.tl_y > 0)
{
split = SANE_FALSE;
}
else
{
split = SANE_TRUE;
}
status = setup_for_scan (dev, settings, split, SANE_FALSE, SANE_FALSE);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "simple_scan: setup_for_scan failed (%s)\n",
sane_strstatus (status));
return status;
}
/* allocate memory fo scan : LINCNT may have been adjusted for CCD reordering */
if (dev->model->is_cis == SANE_TRUE)
{
lines = gl646_get_triple_reg (dev->reg, REG_LINCNT) / 3;
}
else
{
lines = gl646_get_triple_reg (dev->reg, REG_LINCNT) + 1;
}
size = lines * settings.pixels;
if (settings.depth == 16)
bpp = 2;
else
bpp = 1;
size *= bpp;
if (settings.scan_mode == SCAN_MODE_COLOR) /* single pass color */
size *= 3;
*data = malloc (size);
if (!*data)
{
DBG (DBG_error,
"simple_scan: failed to allocate %d bytes of memory\n", size);
return SANE_STATUS_NO_MEM;
}
DBG (DBG_io, "simple_scan: allocated %d bytes of memory for %d lines\n",
size, lines);
/* put back real line number in settings */
settings.lines = lines;
/* initialize frontend */
status = gl646_set_fe (dev, AFE_SET, settings.xres);
if (status != SANE_STATUS_GOOD)
{
free (*data);
DBG (DBG_error,
"simple_scan: failed to set frontend: %s\n",
sane_strstatus (status));
return status;
}
/* no shading correction and not watch dog for simple scan */
dev->reg[reg_0x01].value &= ~(REG01_DVDSET | REG01_DOGENB);
if (shading == SANE_TRUE)
{
dev->reg[reg_0x01].value |= REG01_DVDSET;
}
/* one table movement for simple scan */
dev->reg[reg_0x02].value &= ~REG02_FASTFED;
if (move == SANE_FALSE)
{
/* clear motor power flag if no move */
dev->reg[reg_0x02].value &= ~REG02_MTRPWR;
/* no automatic go home if no movement */
dev->reg[reg_0x02].value &= ~REG02_AGOHOME;
}
if (forward == SANE_FALSE)
{
dev->reg[reg_0x02].value |= REG02_MTRREV;
}
else
{
dev->reg[reg_0x02].value &= ~REG02_MTRREV;
}
/* no automatic go home when using XPA */
if (settings.scan_method == SCAN_METHOD_TRANSPARENCY)
{
dev->reg[reg_0x02].value &= ~REG02_AGOHOME;
}
/* write scan registers */
status = gl646_bulk_write_register (dev, dev->reg,
sizeof (dev->reg) /
sizeof (dev->reg[0]));
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"simple_scan: failed to bulk write registers: %s\n",
sane_strstatus (status));
free (data);
return status;
}
/* starts scan */
status = gl646_begin_scan (dev, dev->reg, move);
if (status != SANE_STATUS_GOOD)
{
free (*data);
DBG (DBG_error, "simple_scan: failed to begin scan: \n");
return status;
}
/* wait for buffers to be filled */
count = 0;
do
{
usleep (10000UL);
RIE (sanei_genesys_get_status (dev, &val));
if (DBG_LEVEL > DBG_info)
{
print_status (val);
}
RIE (sanei_genesys_test_buffer_empty (dev, &empty));
count++;
}
while (empty && count < 1000);
if (count == 1000)
{
free (*data);
DBG (DBG_error, "simple_scan: failed toread data\n");
return SANE_STATUS_IO_ERROR;
}
/* 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,
"simple_scan: failed to read data: %s\n", sane_strstatus (status));
return status;
}
/* in case of CIS scanner, we must reorder data */
if (dev->model->is_cis == SANE_TRUE
&& settings.scan_mode == SCAN_MODE_COLOR)
{
/* alloc one line sized working buffer */
buffer = (unsigned char *) malloc (settings.pixels * 3 * bpp);
if (buffer == NULL)
{
DBG (DBG_error,
"simple_scan: failed to allocate %d bytes of memory\n",
settings.pixels * 3);
return SANE_STATUS_NO_MEM;
}
/* reorder one line of data and put it back to buffer */
if (bpp == 1)
{
for (y = 0; y < lines; y++)
{
/* reorder line */
for (x = 0; x < settings.pixels; x++)
{
buffer[x * 3] = (*data)[y * settings.pixels * 3 + x];
buffer[x * 3 + 1] =
(*data)[y * settings.pixels * 3 + settings.pixels + x];
buffer[x * 3 + 2] =
(*data)[y * settings.pixels * 3 + 2 * settings.pixels +
x];
}
/* copy line back */
memcpy ((*data) + settings.pixels * 3 * y, buffer,
settings.pixels * 3);
}
}
else
{
for (y = 0; y < lines; y++)
{
/* reorder line */
for (x = 0; x < settings.pixels; x++)
{
buffer[x * 6] = (*data)[y * settings.pixels * 6 + x * 2];
buffer[x * 6 + 1] =
(*data)[y * settings.pixels * 6 + x * 2 + 1];
buffer[x * 6 + 2] =
(*data)[y * settings.pixels * 6 + 2 * settings.pixels +
x * 2];
buffer[x * 6 + 3] =
(*data)[y * settings.pixels * 6 + 2 * settings.pixels +
x * 2 + 1];
buffer[x * 6 + 4] =
(*data)[y * settings.pixels * 6 + 4 * settings.pixels +
x * 2];
buffer[x * 6 + 5] =
(*data)[y * settings.pixels * 6 + 4 * settings.pixels +
x * 2 + 1];
}
/* copy line back */
memcpy ((*data) + settings.pixels * 6 * y, buffer,
settings.pixels * 6);
}
}
free (buffer);
}
/* end scan , waiting the motor to stop if needed (if moving), but without ejecting doc */
status = end_scan (dev, dev->reg, SANE_TRUE, SANE_FALSE);
if (status != SANE_STATUS_GOOD)
{
free (*data);
DBG (DBG_error,
"simple_scan: failed to end scan: %s\n", sane_strstatus (status));
return status;
}
DBG (DBG_proc, "simple_scan: end\n");
return status;
}
/**
* Does a simple move of the given distance by doing a scan at lowest resolution
* shading correction. Memory for data is allocated in this function
* and must be freed by caller.
* @param dev device of the scanner
* @param distance distance to move in MM
*/
#ifndef UNIT_TESTING
static
#endif
SANE_Status
simple_move (Genesys_Device * dev, SANE_Int distance)
{
SANE_Status status = SANE_STATUS_INVAL;
unsigned char *data = NULL;
Genesys_Settings settings;
DBG (DBG_proc, "simple_move: %d mm\n", distance);
/* TODO give a no AGOHOME flag */
settings.scan_method = SCAN_METHOD_TRANSPARENCY;
settings.scan_mode = SCAN_MODE_COLOR;
settings.xres = get_lowest_resolution (dev->model->ccd_type, SANE_TRUE);
settings.yres = settings.xres;
settings.tl_y = 0;
settings.tl_x = 0;
settings.pixels =
(dev->sensor.sensor_pixels * settings.xres) / dev->sensor.optical_res;
settings.lines = (distance * settings.xres) / MM_PER_INCH;
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
status =
simple_scan (dev, settings, SANE_TRUE, SANE_TRUE, SANE_FALSE, &data);
free (data);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "simple_move: simple_scan failed\n");
return status;
}
DBG (DBG_proc, "simple_move: end.\n");
return status;
}
/**
* update the status of the required sensor in the scanner session
* the last_val fileds are used to make events 'sticky'
*/
static SANE_Status
gl646_update_hardware_sensors (Genesys_Scanner * session)
{
Genesys_Device *dev = session->dev;
uint8_t value;
SANE_Status status;
/* do what is needed to get a new set of events, but try to not loose
any of them.
*/
status = gl646_gpio_read (dev->dn, &value);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error,
"gl646_update_hardware_sensors: failed to read GPIO %s\n",
sane_strstatus (status));
return status;
}
DBG (DBG_io, "gl646_update_hardware_sensors: GPIO=0x%02x\n", value);
/* scan button */
if ((dev->model->buttons & GENESYS_HAS_SCAN_SW)
&& session->val[OPT_SCAN_SW].b == session->last_val[OPT_SCAN_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_XP200:
session->val[OPT_SCAN_SW].b = ((value & 0x02) != 0);
break;
case GPO_5345:
session->val[OPT_SCAN_SW].b = (value == 0x16);
break;
case GPO_HP2300:
session->val[OPT_SCAN_SW].b = (value == 0x6c);
break;
case GPO_HP3670:
case GPO_HP2400:
session->val[OPT_SCAN_SW].b = ((value & 0x20) == 0);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* email button */
if ((dev->model->buttons & GENESYS_HAS_EMAIL_SW)
&& session->val[OPT_EMAIL_SW].b == session->last_val[OPT_EMAIL_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_5345:
session->val[OPT_EMAIL_SW].b = (value == 0x12);
break;
case GPO_HP3670:
case GPO_HP2400:
session->val[OPT_EMAIL_SW].b = ((value & 0x08) == 0);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* copy button */
if ((dev->model->buttons & GENESYS_HAS_COPY_SW)
&& session->val[OPT_COPY_SW].b == session->last_val[OPT_COPY_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_5345:
session->val[OPT_COPY_SW].b = (value == 0x11);
break;
case GPO_HP2300:
session->val[OPT_COPY_SW].b = (value == 0x5c);
break;
case GPO_HP3670:
case GPO_HP2400:
session->val[OPT_COPY_SW].b = ((value & 0x10) == 0);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* power button */
if ((dev->model->buttons & GENESYS_HAS_POWER_SW)
&& session->val[OPT_POWER_SW].b == session->last_val[OPT_POWER_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_5345:
session->val[OPT_POWER_SW].b = (value == 0x14);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* ocr button */
if ((dev->model->buttons & GENESYS_HAS_OCR_SW)
&& session->val[OPT_OCR_SW].b == session->last_val[OPT_OCR_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_5345:
session->val[OPT_OCR_SW].b = (value == 0x13);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* document detection */
if ((dev->model->buttons & GENESYS_HAS_PAGE_LOADED_SW)
&& session->val[OPT_PAGE_LOADED_SW].b ==
session->last_val[OPT_PAGE_LOADED_SW].b)
{
switch (dev->model->gpo_type)
{
case GPO_XP200:
session->val[OPT_PAGE_LOADED_SW].b = ((value & 0x04) != 0);
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
/* XPA detection */
if (dev->model->flags & GENESYS_FLAG_XPA)
{
switch (dev->model->gpo_type)
{
case GPO_HP3670:
case GPO_HP2400:
/* test if XPA is plugged-in */
if ((value & 0x40) == 0)
{
DBG (DBG_io, "gl646_update_hardware_sensors: enabling XPA\n");
session->opt[OPT_SOURCE].cap &= ~SANE_CAP_INACTIVE;
}
else
{
DBG (DBG_io, "gl646_update_hardware_sensors: disabling XPA\n");
session->opt[OPT_SOURCE].cap |= SANE_CAP_INACTIVE;
}
break;
default:
return SANE_STATUS_UNSUPPORTED;
}
}
return status;
}
static SANE_Status
write_control (Genesys_Device * dev, int resolution)
{
SANE_Status status;
uint8_t control[4];
uint32_t addr = 0xdead;
/* 2300 does not write to 'control' */
if (dev->model->motor_type == MOTOR_HP2300)
return SANE_STATUS_GOOD;
/* MD6471/G2410/HP2300 and XP200 read/write data from an undocumented memory area which
* is after the second slope table */
switch (dev->sensor.optical_res)
{
case 600:
addr = 0x08200;
break;
case 1200:
addr = 0x10200;
break;
case 2400:
addr = 0x1fa00;
break;
default:
DBG (DBG_error, "write_control: failed to compute control address\n");
return SANE_STATUS_INVAL;
}
/* XP200 sets dpi, what other scanner put is unknown yet */
switch (dev->model->motor_type)
{
case MOTOR_XP200:
/* we put scan's dpi, not motor one */
control[0] = LOBYTE (resolution);
control[1] = HIBYTE (resolution);
control[2] = dev->control[4];
control[3] = dev->control[5];
break;
case MOTOR_HP3670:
case MOTOR_HP2400:
case MOTOR_5345:
default:
control[0] = dev->control[2];
control[1] = dev->control[3];
control[2] = dev->control[4];
control[3] = dev->control[5];
break;
}
DBG (DBG_info,
"write_control: control write=0x%02x 0x%02x 0x%02x 0x%02x\n",
control[0], control[1], control[2], control[3]);
status = sanei_genesys_set_buffer_address (dev, addr);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "write_control: failed to set up control address\n");
return SANE_STATUS_INVAL;
}
status = gl646_bulk_write_data (dev, 0x3c, control, 4);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "write_control: failed to set up control\n");
return SANE_STATUS_INVAL;
}
return status;
}
/**
* check if a stored calibration is compatible with requested scan.
* @return SANE_STATUS_GOOD if compatible, SANE_STATUS_UNSUPPORTED if not.
* Whenever an error is met, it is returned.
* @param dev scanner device
* @param cache cache entry to test
* @for_overwrite reserved for future use ...
*/
static SANE_Status
gl646_is_compatible_calibration (Genesys_Device * dev,
Genesys_Calibration_Cache * cache,
int for_overwrite)
{
#ifdef HAVE_SYS_TIME_H
struct timeval time;
#endif
int compatible = 1;
DBG (DBG_proc,
"gl646_is_compatible_calibration: start (for_overwrite=%d)\n",
for_overwrite);
if (cache == NULL)
return SANE_STATUS_UNSUPPORTED;
/* build minimal current_setup for calibration cache use only, it will be better
* computed when during setup for scan
*/
if (dev->settings.scan_mode == SCAN_MODE_COLOR)
{
dev->current_setup.channels = 3;
}
else
{
dev->current_setup.channels = 1;
}
dev->current_setup.xres = dev->settings.xres;
dev->current_setup.scan_method = dev->settings.scan_method;
DBG (DBG_io,
"gl646_is_compatible_calibration: requested=(%d,%f), tested=(%d,%f)\n",
dev->current_setup.channels, dev->current_setup.xres,
cache->used_setup.channels, cache->used_setup.xres);
/* a calibration cache is compatible if color mode and x dpi match the user
* requested scan. In the case of CIS scanners, dpi isn't a criteria */
if (dev->model->is_cis == SANE_FALSE)
{
compatible =
((dev->current_setup.channels == cache->used_setup.channels)
&& (((int) dev->current_setup.xres) ==
((int) cache->used_setup.xres)));
}
else
{
compatible =
(dev->current_setup.channels == cache->used_setup.channels);
}
if (dev->current_setup.scan_method != cache->used_setup.scan_method)
{
DBG (DBG_io,
"gl646_is_compatible_calibration: current method=%d, used=%d\n",
dev->current_setup.scan_method, cache->used_setup.scan_method);
compatible = 0;
}
if (!compatible)
{
DBG (DBG_proc,
"gl646_is_compatible_calibration: completed, non compatible cache\n");
return SANE_STATUS_UNSUPPORTED;
}
/* a cache entry expires after 30 minutes for non sheetfed scanners */
#ifdef HAVE_SYS_TIME_H
gettimeofday (&time, NULL);
if ((time.tv_sec - cache->last_calibration > 30 * 60)
&& (dev->model->is_sheetfed == SANE_FALSE))
{
DBG (DBG_proc,
"gl646_is_compatible_calibration: expired entry, non compatible cache\n");
return SANE_STATUS_UNSUPPORTED;
}
#endif
DBG (DBG_proc,
"gl646_is_compatible_calibration: completed, cache compatible\n");
return SANE_STATUS_GOOD;
}
/**
* search for a full width black or white strip.
* @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
gl646_search_strip (Genesys_Device * dev, SANE_Bool forward, SANE_Bool black)
{
SANE_Status status = SANE_STATUS_GOOD;
SANE_Bool half_ccd = SANE_FALSE;
Genesys_Settings settings;
int res = get_closest_resolution (dev->model->ccd_type, 75, SANE_FALSE);
unsigned char *data = NULL;
unsigned int pass, count, found, x, y;
char title[80];
DBG (DBG_proc, "gl646_search_strip: start\n");
/* adapt to half_ccd case */
if (dev->model->flags & GENESYS_FLAG_HALF_CCD_MODE)
{
/* walk the master mode list to find if half_ccd */
if (is_half_ccd (dev->model->ccd_type, res, SANE_TRUE) == SANE_TRUE)
{
half_ccd = SANE_TRUE;
}
}
/* we set up for a lowest available resolution color grey scan, full width */
settings.scan_method = SCAN_METHOD_FLATBED;
settings.scan_mode = SCAN_MODE_GRAY;
settings.xres = res;
settings.yres = res;
settings.tl_x = 0;
settings.tl_y = 0;
settings.pixels = (SANE_UNFIX (dev->model->x_size) * res) / MM_PER_INCH;
if (half_ccd == SANE_TRUE)
{
settings.pixels /= 2;
}
/* 15 mm at at time */
settings.lines = (15 * settings.yres) / MM_PER_INCH; /* may become a parameter from genesys_devices.c */
settings.depth = 8;
settings.color_filter = 0;
settings.disable_interpolation = 0;
settings.threshold = 0;
settings.exposure_time = 0;
/* signals if a strip of the given color has been found */
found = 0;
/* detection pass done */
pass = 0;
/* loop until strip is found or maximum pass number done */
while (pass < 20 && !found)
{
/* scan a full width strip */
status =
simple_scan (dev, settings, SANE_TRUE, forward, SANE_FALSE, &data);
if (status != SANE_STATUS_GOOD)
{
DBG (DBG_error, "gl646_search_strip: simple_scan failed\n");
free (data);
return status;
}
if (DBG_LEVEL >= DBG_data)
{
sprintf (title, "search_strip_%s%02d.pnm", forward ? "fwd" : "bwd",
pass);
sanei_genesys_write_pnm_file (title, data, settings.depth, 1,
settings.pixels, settings.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 < settings.lines && !found; y++)
{
count = 0;
/* count of white/black pixels depending on the color searched */
for (x = 0; x < settings.pixels; x++)
{
/* when searching for black, detect white pixels */
if (black && data[y * settings.pixels + x] > 90)
{
count++;
}
/* when searching for white, detect black pixels */
if (!black && data[y * settings.pixels + x] < 60)
{
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) / settings.pixels < 3)
{
found = 1;
DBG (DBG_data,
"gl646_search_strip: strip found forward during pass %d at line %d\n",
pass, y);
}
else
{
DBG (DBG_data, "gl646_search_strip: pixels=%d, count=%d\n",
settings.pixels, count);
}
}
}
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 < settings.lines; y++)
{
/* count of white/black pixels depending on the color searched */
for (x = 0; x < settings.pixels; x++)
{
/* when searching for black, detect white pixels */
if (black && data[y * settings.pixels + x] > 60)
{
count++;
}
/* when searching for white, detect black pixels */
if (!black && data[y * settings.pixels + x] < 60)
{
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) / (settings.pixels * settings.lines) < 3)
{
found = 1;
DBG (DBG_data,
"gl646_search_strip: strip found backward during pass %d \n",
pass);
}
else
{
DBG (DBG_data, "gl646_search_strip: pixels=%d, count=%d\n",
settings.pixels, count);
}
}
pass++;
}
free (data);
if (found)
{
status = SANE_STATUS_GOOD;
DBG (DBG_info, "gl646_search_strip: strip found\n");
}
else
{
status = SANE_STATUS_UNSUPPORTED;
DBG (DBG_info, "gl646_search_strip: strip not found\n");
}
return status;
}
/** the gl646 command set */
static Genesys_Command_Set gl646_cmd_set = {
"gl646-generic", /* the name of this set */
gl646_init,
gl646_init_regs_for_warmup,
gl646_init_regs_for_coarse_calibration,
gl646_init_regs_for_shading,
gl646_init_regs_for_scan,
gl646_get_filter_bit,
gl646_get_lineart_bit,
gl646_get_bitset_bit,
gl646_get_gain4_bit,
gl646_get_fast_feed_bit,
gl646_test_buffer_empty_bit,
gl646_test_motor_flag_bit,
gl646_bulk_full_size,
gl646_public_set_fe,
gl646_set_powersaving,
gl646_save_power,
gl646_set_motor_power,
gl646_set_lamp_power,
gl646_begin_scan,
gl646_end_scan,
gl646_send_gamma_table,
gl646_search_start_position,
gl646_offset_calibration,
gl646_coarse_gain_calibration,
gl646_led_calibration,
gl646_slow_back_home,
gl646_bulk_write_register,
gl646_bulk_write_data,
gl646_bulk_read_data,
gl646_update_hardware_sensors,
/* sheetfed related functions */
gl646_load_document,
gl646_detect_document_end,
gl646_eject_document,
gl646_search_strip,
gl646_is_compatible_calibration,
gl646_move_to_ta,
NULL
};
SANE_Status
sanei_gl646_init_cmd_set (Genesys_Device * dev)
{
dev->model->cmd_set = &gl646_cmd_set;
return SANE_STATUS_GOOD;
}
/* vim: set sw=2 cino=>2se-1sn-1s{s^-1st0(0u0 smarttab expandtab: */