kopia lustrzana https://gitlab.com/sane-project/backends
870 wiersze
23 KiB
C
870 wiersze
23 KiB
C
/* sane - Scanner Access Now Easy.
|
|
|
|
Copyright (C) 2007-2013 stef.dev@free.fr
|
|
|
|
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.
|
|
*/
|
|
|
|
/* this file contains all the low level functions needed for the higher level
|
|
* functions of the sane standard. They are put there to keep files smaller
|
|
* and separate functions with different goals.
|
|
*/
|
|
|
|
#include "../include/sane/config.h"
|
|
#include "../include/sane/sane.h"
|
|
#include "../include/sane/sanei_backend.h"
|
|
#include "../include/sane/sanei_usb.h"
|
|
|
|
#include <stdio.h>
|
|
#ifdef HAVE_SYS_TIME_H
|
|
#include <sys/time.h>
|
|
#endif
|
|
#include "rts8891_low.h"
|
|
|
|
#define RTS8891_BUILD 30
|
|
#define RTS8891_MAX_REGISTERS 244
|
|
|
|
/* init rts8891 library */
|
|
static void
|
|
rts8891_low_init (void)
|
|
{
|
|
DBG_INIT ();
|
|
DBG (DBG_info, "RTS8891 low-level functions, version %d.%d-%d\n",
|
|
SANE_CURRENT_MAJOR, V_MINOR, RTS8891_BUILD);
|
|
}
|
|
|
|
|
|
/****************************************************************/
|
|
/* ASIC specific functions */
|
|
/****************************************************************/
|
|
|
|
/* write all registers, taking care of the special 0xaa value which
|
|
* must be escaped with a zero
|
|
*/
|
|
static SANE_Status
|
|
rts8891_write_all (SANE_Int devnum, SANE_Byte * regs, SANE_Int count)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Byte local_regs[RTS8891_MAX_REGISTERS];
|
|
size_t size = 0;
|
|
SANE_Byte buffer[260];
|
|
unsigned int i, j;
|
|
char message[256 * 5];
|
|
|
|
if (DBG_LEVEL > DBG_io)
|
|
{
|
|
for (i = 0; i < (unsigned int) count; i++)
|
|
{
|
|
if (i != 0xb3)
|
|
sprintf (message + 5 * i, "0x%02x ", regs[i]);
|
|
else
|
|
sprintf (message + 5 * i, "---- ");
|
|
}
|
|
DBG (DBG_io, "rts8891_write_all : write_all(0x00,%d)=%s\n", count,
|
|
message);
|
|
}
|
|
|
|
/* copy register set and escaping 0xaa values */
|
|
/* b0, b1 abd b3 values may be scribled, but that isn't important */
|
|
/* since they are read-only registers */
|
|
j = 0;
|
|
for (i = 0; i < 0xb3; i++)
|
|
{
|
|
local_regs[j] = regs[i];
|
|
if (local_regs[j] == 0xaa && i < 0xb3)
|
|
{
|
|
j++;
|
|
local_regs[j] = 0x00;
|
|
}
|
|
j++;
|
|
}
|
|
buffer[0] = 0x88;
|
|
buffer[1] = 0;
|
|
buffer[2] = 0x00;
|
|
buffer[3] = 0xb3;
|
|
for (i = 0; i < j; i++)
|
|
buffer[i + 4] = local_regs[i];
|
|
/* the USB block is size + 4 bytes of header long */
|
|
size = j + 4;
|
|
if (sanei_usb_write_bulk (devnum, buffer, &size) != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error,
|
|
"rts88xx_write_all : write registers part 1 failed ...\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
|
|
size = count - 0xb4; /* we need to substract one reg since b3 won't be written */
|
|
buffer[0] = 0x88;
|
|
buffer[1] = 0xb4;
|
|
buffer[2] = 0x00;
|
|
buffer[3] = size;
|
|
for (i = 0; i < size; i++)
|
|
buffer[i + 4] = regs[0xb4 + i];
|
|
/* the USB block is size + 4 bytes of header long */
|
|
size += 4;
|
|
if (sanei_usb_write_bulk (devnum, buffer, &size) != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error,
|
|
"rts88xx_write_all : write registers part 2 failed ...\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
|
|
/* this functions "commits" pending scan command */
|
|
static SANE_Status
|
|
rts8891_commit (SANE_Int devnum, SANE_Byte value)
|
|
{
|
|
SANE_Status status;
|
|
SANE_Byte reg;
|
|
|
|
reg = value;
|
|
sanei_rts88xx_write_reg (devnum, 0xd3, ®);
|
|
sanei_rts88xx_cancel (devnum);
|
|
sanei_rts88xx_write_control (devnum, 0x08);
|
|
status = sanei_rts88xx_write_control (devnum, 0x08);
|
|
return status;
|
|
}
|
|
|
|
/* this functions reads button status */
|
|
static SANE_Status
|
|
rts8891_read_buttons (SANE_Int devnum, SANE_Int * mask)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Byte reg;
|
|
|
|
/* check CONTROL_REG */
|
|
sanei_rts88xx_read_reg (devnum, CONTROL_REG, ®);
|
|
|
|
/* read 'base' button status */
|
|
sanei_rts88xx_read_reg (devnum, 0x25, ®);
|
|
DBG (DBG_io, "rts8891_read_buttons: r25=0x%02x\n", reg);
|
|
*mask |= reg;
|
|
|
|
/* read 'extended' button status */
|
|
sanei_rts88xx_read_reg (devnum, 0x1a, ®);
|
|
DBG (DBG_io, "rts8891_read_buttons: r1a=0x%02x\n", reg);
|
|
*mask |= reg << 8;
|
|
|
|
/* clear register r25 */
|
|
reg = 0x00;
|
|
sanei_rts88xx_write_reg (devnum, 0x25, ®);
|
|
|
|
/* clear register r1a */
|
|
sanei_rts88xx_read_reg (devnum, 0x1a, ®);
|
|
reg = 0x00;
|
|
status = sanei_rts88xx_write_reg (devnum, 0x1a, ®);
|
|
|
|
DBG (DBG_info, "rts8891_read_buttons: mask=0x%04x\n", *mask);
|
|
return status;
|
|
}
|
|
|
|
/*
|
|
* Does a simple scan based on the given register set, returning data in a
|
|
* preallocated buffer of the claimed size.
|
|
* sanei_rts88xx_data_count cannot be made reliable, when the announced data
|
|
* amount is read, it may no be ready, leading to errors. To work around
|
|
* it, we read data count one more time before reading.
|
|
*/
|
|
static SANE_Status
|
|
rts8891_simple_scan (SANE_Int devnum, SANE_Byte * regs, int regcount,
|
|
SANE_Int format, SANE_Word total, unsigned char *image)
|
|
{
|
|
SANE_Word count, read, len, dummy;
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Byte control;
|
|
|
|
rts8891_write_all (devnum, regs, regcount);
|
|
rts8891_commit (devnum, format);
|
|
|
|
read = 0;
|
|
count = 0;
|
|
while (count == 0)
|
|
{
|
|
status = sanei_rts88xx_data_count (devnum, &count);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "simple_scan: failed to wait for data\n");
|
|
return status;
|
|
}
|
|
if (count == 0)
|
|
{
|
|
status = sanei_rts88xx_read_reg (devnum, CONTROL_REG, &control);
|
|
if (((control & 0x08) == 0) || (status != SANE_STATUS_GOOD))
|
|
{
|
|
DBG (DBG_error, "simple_scan: failed to wait for data\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* data reading */
|
|
read = 0;
|
|
while ((read < total) && (count != 0 || (control & 0x08) == 0x08))
|
|
{
|
|
/* sync ? */
|
|
status = sanei_rts88xx_data_count (devnum, &dummy);
|
|
|
|
/* read */
|
|
if (count > 0)
|
|
{
|
|
len = count;
|
|
/* read even size unless last chunk */
|
|
if ((len & 1) && (read + len < total))
|
|
{
|
|
len++;
|
|
}
|
|
if (len > RTS88XX_MAX_XFER_SIZE)
|
|
{
|
|
len = RTS88XX_MAX_XFER_SIZE;
|
|
}
|
|
if (len > 0)
|
|
{
|
|
status = sanei_rts88xx_read_data (devnum, &len, image + read);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error,
|
|
"simple_scan: failed to read from scanner\n");
|
|
return status;
|
|
}
|
|
read += len;
|
|
}
|
|
}
|
|
|
|
/* don't try to read data count if we have enough data */
|
|
if (read < total)
|
|
{
|
|
status = sanei_rts88xx_data_count (devnum, &count);
|
|
}
|
|
else
|
|
{
|
|
count = 0;
|
|
}
|
|
if (count == 0)
|
|
{
|
|
sanei_rts88xx_read_reg (devnum, CONTROL_REG, &control);
|
|
}
|
|
}
|
|
|
|
/* sanity check */
|
|
if (read < total)
|
|
{
|
|
DBG (DBG_io2, "simple_scan: ERROR, %d bytes missing ... \n",
|
|
total - read);
|
|
}
|
|
|
|
/* wait for the motor to stop */
|
|
do
|
|
{
|
|
sanei_rts88xx_read_reg (devnum, CONTROL_REG, &control);
|
|
}
|
|
while ((control & 0x08) == 0x08);
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* set the data format. Is part of the commit sequence. Then returned
|
|
* value is the value used in d3 register for a scan.
|
|
*/
|
|
static SANE_Int
|
|
rts8891_data_format (SANE_Int dpi, int sensor)
|
|
{
|
|
SANE_Byte reg = 0x00;
|
|
|
|
/* it seems that lower nibble is a divisor */
|
|
if (sensor == SENSOR_TYPE_BARE || sensor == SENSOR_TYPE_XPA)
|
|
{
|
|
switch (dpi)
|
|
{
|
|
case 75:
|
|
reg = 0x02;
|
|
break;
|
|
case 150:
|
|
if (sensor == SENSOR_TYPE_BARE)
|
|
reg = 0x0e;
|
|
else
|
|
reg = 0x0b;
|
|
break;
|
|
case 300:
|
|
reg = 0x17;
|
|
break;
|
|
case 600:
|
|
if (sensor == SENSOR_TYPE_BARE)
|
|
reg = 0x02;
|
|
else
|
|
reg = 0x0e;
|
|
break;
|
|
case 1200:
|
|
if (sensor == SENSOR_TYPE_BARE)
|
|
reg = 0x17;
|
|
else
|
|
reg = 0x05;
|
|
break;
|
|
}
|
|
}
|
|
if (sensor == SENSOR_TYPE_4400 || sensor == SENSOR_TYPE_4400_BARE)
|
|
{
|
|
switch (dpi)
|
|
{
|
|
case 75:
|
|
reg = 0x02;
|
|
break;
|
|
case 150:
|
|
if (sensor == SENSOR_TYPE_4400)
|
|
reg = 0x0b;
|
|
else
|
|
reg = 0x17;
|
|
break;
|
|
case 300:
|
|
reg = 0x17;
|
|
break;
|
|
case 600:
|
|
if (sensor == SENSOR_TYPE_4400)
|
|
reg = 0x0e;
|
|
else
|
|
reg = 0x02;
|
|
break;
|
|
case 1200:
|
|
if (sensor == SENSOR_TYPE_4400)
|
|
reg = 0x05;
|
|
else
|
|
reg = 0x17;
|
|
break;
|
|
}
|
|
}
|
|
return reg;
|
|
}
|
|
|
|
/**
|
|
* set up default values for a 75xdpi, 150 ydpi scan
|
|
*/
|
|
static void
|
|
rts8891_set_default_regs (SANE_Byte * scanner_regs)
|
|
{
|
|
SANE_Byte default_75[RTS8891_MAX_REGISTERS] =
|
|
{ 0xe5, 0x41, 0x1f, 0x1f, 0x1f, 0x1f, 0x1f, 0x1f, 0x0a, 0x0a, 0x0a, 0x70,
|
|
0x00, 0x00, 0x00, 0x00, 0x28, 0x3f, 0xff, 0x20, 0xf8, 0x28, 0x07, 0x00,
|
|
0xff, 0x00, 0x00, 0x00, 0x00, 0x20, 0x00,
|
|
0x00, 0x3a, 0xf2, 0x00, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x10, 0x00, 0x07, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0x8c,
|
|
0x76, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xe1, 0x14, 0x18, 0x15, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0xff, 0x3f, 0x80, 0x68, 0x00, 0x00,
|
|
0x00, 0x00, 0x01, 0x0e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xcc,
|
|
0x27, 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x0f, 0x00,
|
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00,
|
|
0x0e, 0x00, 0x00, 0xf0, 0xff, 0xf5, 0xf7, 0xea, 0x0b, 0x03, 0x05, 0x86,
|
|
0x1b, 0x30, 0xf6, 0xa2, 0x27, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x14, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00
|
|
};
|
|
unsigned int i;
|
|
for (i = 0; i < RTS8891_MAX_REGISTERS; i++)
|
|
scanner_regs[i] = default_75[i];
|
|
}
|
|
|
|
static SANE_Status
|
|
rts8891_move (struct Rts8891_Device *device, SANE_Byte * regs,
|
|
SANE_Int distance, SANE_Bool forward)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Byte regs10, regs11;
|
|
|
|
DBG (DBG_proc, "rts8891_move: start\n");
|
|
DBG (DBG_io, "rts8891_move: %d lines %s, sensor=%d\n", distance,
|
|
forward == SANE_TRUE ? "forward" : "backward", device->sensor);
|
|
|
|
/* prepare scan */
|
|
rts8891_set_default_regs (regs);
|
|
if (device->sensor != SENSOR_TYPE_4400
|
|
&& device->sensor != SENSOR_TYPE_4400_BARE)
|
|
{
|
|
regs10 = 0x20;
|
|
regs11 = 0x28;
|
|
}
|
|
else
|
|
{
|
|
regs10 = 0x10;
|
|
regs11 = 0x2a;
|
|
}
|
|
|
|
regs[0x32] = 0x80;
|
|
regs[0x33] = 0x81;
|
|
regs[0x35] = 0x10;
|
|
regs[0x36] = 0x24;
|
|
regs[0x39] = 0x02;
|
|
regs[0x3a] = 0x0e;
|
|
|
|
regs[0x64] = 0x01;
|
|
regs[0x65] = 0x20;
|
|
|
|
regs[0x79] = 0x20;
|
|
regs[0x7a] = 0x01;
|
|
|
|
regs[0x80] = 0x32;
|
|
regs[0x82] = 0x33;
|
|
regs[0x85] = 0x46;
|
|
regs[0x86] = 0x0b;
|
|
regs[0x87] = 0x8c;
|
|
regs[0x88] = 0x10;
|
|
regs[0x89] = 0xb2;
|
|
regs[0x8d] = 0x3b;
|
|
regs[0x8e] = 0x60;
|
|
|
|
regs[0x90] = 0x1c;
|
|
regs[0xb2] = 0x16; /* 0x10 : stop when at home, 0x04: no data */
|
|
regs[0xc0] = 0x00;
|
|
regs[0xc1] = 0x00;
|
|
regs[0xc3] = 0x00;
|
|
regs[0xc4] = 0x00;
|
|
regs[0xc5] = 0x00;
|
|
regs[0xc6] = 0x00;
|
|
regs[0xc7] = 0x00;
|
|
regs[0xc8] = 0x00;
|
|
regs[0xca] = 0x00;
|
|
regs[0xcd] = 0x00;
|
|
regs[0xce] = 0x00;
|
|
regs[0xcf] = 0x00;
|
|
regs[0xd0] = 0x00;
|
|
regs[0xd1] = 0x00;
|
|
regs[0xd2] = 0x00;
|
|
regs[0xd3] = 0x00;
|
|
regs[0xd4] = 0x00;
|
|
regs[0xd6] = 0x6b;
|
|
regs[0xd7] = 0x00;
|
|
regs[0xd8] = 0x00;
|
|
regs[0xd9] = 0xad;
|
|
regs[0xda] = 0xa7;
|
|
regs[0xe2] = 0x17;
|
|
regs[0xe3] = 0x0d;
|
|
regs[0xe4] = 0x06;
|
|
regs[0xe5] = 0xf9;
|
|
regs[0xe7] = 0x53;
|
|
regs[0xe8] = 0x02;
|
|
regs[0xe9] = 0x02;
|
|
|
|
/* hp4400 sensors */
|
|
if (device->sensor == SENSOR_TYPE_4400
|
|
|| device->sensor == SENSOR_TYPE_4400_BARE)
|
|
{
|
|
regs[0x13] = 0x39; /* 0x20 */
|
|
regs[0x14] = 0xf0; /* 0xf8 */
|
|
regs[0x15] = 0x29; /* 0x28 */
|
|
regs[0x16] = 0x0f; /* 0x07 */
|
|
regs[0x17] = 0x10; /* 0x00 */
|
|
regs[0x23] = 0x00; /* 0xff */
|
|
regs[0x35] = 0x29; /* 0x10 */
|
|
regs[0x36] = 0x21; /* 0x24 */
|
|
regs[0x39] = 0x00; /* 0x02 */
|
|
regs[0x80] = 0xb0; /* 0x32 */
|
|
regs[0x82] = 0xb1; /* 0x33 */
|
|
regs[0xe2] = 0x0b; /* 0x17 */
|
|
regs[0xe5] = 0xf3; /* 0xf9 */
|
|
regs[0xe6] = 0x01; /* 0x00 */
|
|
}
|
|
|
|
/* disable CCD */
|
|
regs[0] = 0xf5;
|
|
|
|
sanei_rts88xx_set_status (device->devnum, regs, regs10, regs11);
|
|
sanei_rts88xx_set_scan_area (regs, distance, distance + 1, 100, 200);
|
|
sanei_rts88xx_set_gain (regs, 16, 16, 16);
|
|
sanei_rts88xx_set_offset (regs, 127, 127, 127);
|
|
|
|
/* forward/backward */
|
|
/* 0x2c is forward, 0x24 backward */
|
|
if (forward == SANE_TRUE)
|
|
{ /* forward */
|
|
regs[0x36] = regs[0x36] | 0x08;
|
|
}
|
|
else
|
|
{ /* backward */
|
|
regs[0x36] = regs[0x36] & 0xf7;
|
|
}
|
|
|
|
/* write regiters values */
|
|
status = rts8891_write_all (device->devnum, regs, RTS8891_MAX_REGISTERS);
|
|
|
|
/* commit it */
|
|
rts8891_commit (device->devnum, 0x00);
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* wait for the scanning head to reach home position
|
|
*/
|
|
static SANE_Status
|
|
rts8891_wait_for_home (struct Rts8891_Device *device, SANE_Byte * regs)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Byte motor, sensor, reg;
|
|
|
|
DBG (DBG_proc, "rts8891_wait_for_home: start\n");
|
|
|
|
/* wait for controller home bit to raise, no timeout */
|
|
/* at each loop we check that motor is on, then that the sensor bit it cleared */
|
|
do
|
|
{
|
|
sanei_rts88xx_read_reg (device->devnum, CONTROL_REG, &motor);
|
|
sanei_rts88xx_read_reg (device->devnum, CONTROLER_REG, &sensor);
|
|
}
|
|
while ((motor & 0x08) && ((sensor & 0x02) == 0));
|
|
|
|
/* flag that device has finished parking */
|
|
device->parking=SANE_FALSE;
|
|
|
|
/* check for error */
|
|
if (((motor & 0x08) == 0x00) && ((sensor & 0x02) == 0))
|
|
{
|
|
DBG (DBG_error,
|
|
"rts8891_wait_for_home: error, motor stopped before head parked\n");
|
|
status = SANE_STATUS_INVAL;
|
|
}
|
|
|
|
/* re-enable CCD */
|
|
regs[0] = regs[0] & 0xef;
|
|
|
|
sanei_rts88xx_cancel (device->devnum);
|
|
|
|
/* reset ? so we don't need to read data */
|
|
reg = 0;
|
|
/* b7: movement on/off, b3-b0 : movement divisor */
|
|
sanei_rts88xx_write_reg (device->devnum, 0x33, ®);
|
|
sanei_rts88xx_write_reg (device->devnum, 0x33, ®);
|
|
/* movement direction */
|
|
sanei_rts88xx_write_reg (device->devnum, 0x36, ®);
|
|
sanei_rts88xx_cancel (device->devnum);
|
|
|
|
DBG (DBG_proc, "rts8891_wait_for_home: end\n");
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* move the head backward by a huge line number then poll home sensor until
|
|
* head has get back home. We have our own copy of the registers to avoid
|
|
* messing scanner status
|
|
*/
|
|
static SANE_Status
|
|
rts8891_park (struct Rts8891_Device *device, SANE_Byte *regs, SANE_Bool wait)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
|
|
DBG (DBG_proc, "rts8891_park: start\n");
|
|
|
|
device->parking=SANE_TRUE;
|
|
rts8891_move (device, regs, 8000, SANE_FALSE);
|
|
|
|
if(wait==SANE_TRUE)
|
|
{
|
|
status = rts8891_wait_for_home (device,regs);
|
|
}
|
|
|
|
DBG (DBG_proc, "rts8891_park: end\n");
|
|
return status;
|
|
}
|
|
|
|
/* reads data from scanner.
|
|
* First we wait for some data to be available and then loop reading
|
|
* from scanner until the required amount is reached.
|
|
* We handle non blocking I/O by returning immediatly (with SANE_STATUS_BUSY)
|
|
* if there is no data available from scanner. But once read is started,
|
|
* all the required amount is read. Once wait for data succeeded, we still poll
|
|
* for data in order no to read it too fast, but we don' take care of non blocking
|
|
* mode since we cope with it on first data wait.
|
|
*/
|
|
static SANE_Status
|
|
read_data (struct Rts8891_Session *session, SANE_Byte * dest, SANE_Int length)
|
|
{
|
|
SANE_Status status = SANE_STATUS_GOOD;
|
|
SANE_Int count, read, len, dummy;
|
|
struct Rts8891_Device *dev = session->dev;
|
|
static FILE *raw = NULL; /* for debugging purpose we need it static */
|
|
SANE_Byte control = 0x08;
|
|
unsigned char buffer[RTS88XX_MAX_XFER_SIZE];
|
|
|
|
DBG (DBG_proc, "read_data: start\n");
|
|
DBG (DBG_proc, "read_data: requiring %d bytes\n", length);
|
|
|
|
/* wait for data being available and handle non blocking mode */
|
|
/* only when data reading hasn't produce any data yet */
|
|
if (dev->read == 0)
|
|
{
|
|
do
|
|
{
|
|
status = sanei_rts88xx_data_count (dev->devnum, &count);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "read_data: failed to wait for data\n");
|
|
return status;
|
|
}
|
|
if (count == 0)
|
|
{
|
|
sanei_rts88xx_read_reg (dev->devnum, CONTROL_REG, &control);
|
|
if ((control & 0x08) == 0 && (count == 0))
|
|
{
|
|
DBG (DBG_error,
|
|
"read_data: scanner stopped being busy before data are available\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
}
|
|
|
|
/* in case there is no data, we return BUSY since this mean */
|
|
/* that scanning head hasn't reach is position and data hasn't */
|
|
/* come yet */
|
|
if (session->non_blocking && count == 0)
|
|
{
|
|
|
|
dev->regs[LAMP_REG] = 0x8d;
|
|
sanei_rts88xx_write_reg (dev->devnum, LAMP_REG,
|
|
&(dev->regs[LAMP_REG]));
|
|
DBG (DBG_io, "read_data: no data vailable\n");
|
|
DBG (DBG_proc, "read_data: end\n");
|
|
return SANE_STATUS_DEVICE_BUSY;
|
|
}
|
|
}
|
|
while (count == 0);
|
|
}
|
|
else
|
|
{ /* start of read for a new block */
|
|
status = sanei_rts88xx_data_count (dev->devnum, &count);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "read_data: failed to wait for data\n");
|
|
return status;
|
|
}
|
|
if (count == 0)
|
|
{
|
|
sanei_rts88xx_read_reg (dev->devnum, CONTROL_REG, &control);
|
|
if ((control & 0x08) == 0 && (count == 0))
|
|
{
|
|
DBG (DBG_error,
|
|
"read_data: scanner stopped being busy before data are available\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* fill scanned data buffer */
|
|
read = 0;
|
|
|
|
/* now loop reading data until we have the amount requested */
|
|
/* we also take care of not reading too much data */
|
|
while (read < length && dev->read < dev->to_read
|
|
&& ((control & 0x08) == 0x08))
|
|
{
|
|
/* used to sync */
|
|
if (dev->read == 0)
|
|
{
|
|
status = sanei_rts88xx_data_count (dev->devnum, &dummy);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "read_data: failed to read data count\n");
|
|
return status;
|
|
}
|
|
}
|
|
|
|
/* if there is data to read, read it */
|
|
if (count > 0)
|
|
{
|
|
len = count;
|
|
|
|
if (len > RTS88XX_MAX_XFER_SIZE)
|
|
{
|
|
len = RTS88XX_MAX_XFER_SIZE;
|
|
}
|
|
|
|
/* we only read even size blocks of data */
|
|
if (len & 1)
|
|
{
|
|
DBG (DBG_io, "read_data: round to next even number\n");
|
|
len++;
|
|
}
|
|
|
|
if (len > length - read)
|
|
{
|
|
len = length - read;
|
|
}
|
|
|
|
status = sanei_rts88xx_read_data (dev->devnum, &len, dest + read);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "read_data: failed to read from scanner\n");
|
|
return status;
|
|
}
|
|
|
|
/* raw data tracing */
|
|
if (DBG_LEVEL >= DBG_io2)
|
|
{
|
|
/* open a new file only when no data scanned */
|
|
if (dev->read == 0)
|
|
{
|
|
raw = fopen ("raw_data.pnm", "wb");
|
|
if (raw != NULL)
|
|
{
|
|
/* PNM header */
|
|
fprintf (raw, "P%c\n%d %d\n255\n",
|
|
session->params.format ==
|
|
SANE_FRAME_RGB
|
|
|| session->emulated_gray ==
|
|
SANE_TRUE ? '6' : '5', dev->pixels,
|
|
dev->lines);
|
|
}
|
|
}
|
|
if (raw != NULL)
|
|
{
|
|
fwrite (dest + read, 1, len, raw);
|
|
}
|
|
}
|
|
|
|
/* move pointer and counter */
|
|
read += len;
|
|
dev->read += len;
|
|
DBG (DBG_io2, "read_data: %d/%d\n", dev->read, dev->to_read);
|
|
}
|
|
|
|
/* in fast scan mode, read data count
|
|
* in slow scan, head moves by the amount of data read */
|
|
status = sanei_rts88xx_data_count (dev->devnum, &count);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_error, "read_data: failed to read data count\n");
|
|
return status;
|
|
}
|
|
|
|
/* if no data, check if still scanning */
|
|
if (count == 0 && dev->read < dev->to_read)
|
|
{
|
|
sanei_rts88xx_read_reg (dev->devnum, CONTROL_REG, &control);
|
|
if ((control & 0x08) == 0x00)
|
|
{
|
|
DBG (DBG_error,
|
|
"read_data: scanner stopped being busy before data are available\n");
|
|
return SANE_STATUS_IO_ERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* end of physical reads */
|
|
if (dev->read >= dev->to_read)
|
|
{
|
|
/* check there is no more data in case of a bug */
|
|
sanei_rts88xx_data_count (dev->devnum, &count);
|
|
if (count > 0)
|
|
{
|
|
DBG (DBG_warn,
|
|
"read_data: %d bytes are still available from scanner\n",
|
|
count);
|
|
|
|
/* flush left-over data */
|
|
while (count > 0)
|
|
{
|
|
len = count;
|
|
if (len > RTS88XX_MAX_XFER_SIZE)
|
|
{
|
|
len = RTS88XX_MAX_XFER_SIZE;
|
|
}
|
|
|
|
/* we only read even size blocks of data */
|
|
if (len & 1)
|
|
{
|
|
len++;
|
|
}
|
|
sanei_rts88xx_read_data (dev->devnum, &len, buffer);
|
|
sanei_rts88xx_data_count (dev->devnum, &count);
|
|
}
|
|
}
|
|
|
|
/* wait for motor to stop at the end of the scan */
|
|
do
|
|
{
|
|
sanei_rts88xx_read_reg (dev->devnum, CONTROL_REG, &control);
|
|
}
|
|
while ((control & 0x08) != 0);
|
|
|
|
/* close log file if needed */
|
|
if (DBG_LEVEL >= DBG_io2)
|
|
{
|
|
if (raw != NULL)
|
|
{
|
|
fclose (raw);
|
|
raw = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
DBG (DBG_io, "read_data: read %d bytes from scanner\n", length);
|
|
DBG (DBG_proc, "read_data: end\n");
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* set scanner idle before leaving xxx_quiet()
|
|
write_reg(0x33,1)=0x00
|
|
write_reg(0x33,1)=0x00
|
|
write_reg(0x36,1)=0x00
|
|
prepare();
|
|
------
|
|
write_reg(LAMP_REG,1)=0x80
|
|
write_reg(LAMP_REG,1)=0xad
|
|
read_reg(0x14,2)=0xf8 0x28
|
|
write_reg(0x14,2)=0x78 0x28
|
|
get_status()=0x20 0x3f
|
|
read_reg(0xb1,1)=0x00
|
|
read_control()=0x00
|
|
reset_lamp()=(0x20,0x3f)
|
|
write_reg(LAMP_REG,1)=0x8d
|
|
write_reg(LAMP_REG,1)=0xad
|
|
*/
|
|
|
|
/* vim: set sw=2 cino=>2se-1sn-1s{s^-1st0(0u0 smarttab expandtab: */
|