kopia lustrzana https://gitlab.com/sane-project/backends
1023 wiersze
26 KiB
C
1023 wiersze
26 KiB
C
/* sane - Scanner Access Now Easy.
|
|
Copyright (C) 2003 Martijn van Oosterhout <kleptog@svana.org>
|
|
Copyright (C) 2003 Thomas Soumarmon <thomas.soumarmon@cogitae.net>
|
|
|
|
Originally copied from HP3300 testtools. Original notice follows:
|
|
|
|
Copyright (C) 2001 Bertrik Sikken (bertrik@zonnet.nl)
|
|
|
|
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.
|
|
*/
|
|
|
|
|
|
/*
|
|
SANE interface for hp54xx scanners. Prototype.
|
|
Parts of this source were inspired by other backends.
|
|
*/
|
|
|
|
#include "../include/sane/config.h"
|
|
|
|
/* definitions for debug */
|
|
#include "hp5400_debug.h"
|
|
|
|
#include "../include/sane/sane.h"
|
|
#include "../include/sane/sanei.h"
|
|
#include "../include/sane/sanei_backend.h"
|
|
#include "../include/sane/sanei_config.h"
|
|
#include "../include/sane/saneopts.h"
|
|
#include "../include/sane/sanei_usb.h"
|
|
|
|
#include <stdlib.h> /* malloc, free */
|
|
#include <string.h> /* memcpy */
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
|
|
#define HP5400_CONFIG_FILE "hp5400.conf"
|
|
|
|
#include "hp5400.h"
|
|
|
|
/* includes for data transfer methods */
|
|
#include "hp5400.h"
|
|
|
|
#ifdef STANDALONE
|
|
#include "hp5400_scanner.h"
|
|
#endif
|
|
|
|
#if defined(LINUX_USB_SUPPORT)
|
|
#include "hp5400_linux.c"
|
|
#endif
|
|
#if defined(USCANNER_SUPPORT)
|
|
#include "hp5400_uscanner.c"
|
|
#endif
|
|
#if defined(LIBUSB_SUPPORT)
|
|
#include "hp5400_libusb.c"
|
|
#endif
|
|
#if defined(LIBIEEE1284_SUPPORT)
|
|
#include "hp5400_ieee1284.c"
|
|
#endif
|
|
|
|
|
|
|
|
/* other definitions */
|
|
#ifndef min
|
|
#define min(A,B) (((A)<(B)) ? (A) : (B))
|
|
#endif
|
|
#ifndef max
|
|
#define max(A,B) (((A)>(B)) ? (A) : (B))
|
|
#endif
|
|
|
|
#define TRUE 1
|
|
#define FALSE 0
|
|
|
|
#define MM_TO_PIXEL(_mm_, _dpi_) ((_mm_) * (_dpi_) / 25.4)
|
|
#define PIXEL_TO_MM(_pixel_, _dpi_) ((_pixel_) * 25.4 / (_dpi_))
|
|
|
|
#define NUM_GAMMA_ENTRIES 65536
|
|
|
|
|
|
/* options enumerator */
|
|
typedef enum
|
|
{
|
|
optCount = 0,
|
|
|
|
optGroupGeometry,
|
|
optTLX, optTLY, optBRX, optBRY,
|
|
optDPI,
|
|
|
|
optGroupImage,
|
|
|
|
optGammaTableRed, /* Gamma Tables */
|
|
optGammaTableGreen,
|
|
optGammaTableBlue,
|
|
|
|
optLast, /* Disable the offset code */
|
|
|
|
optGroupMisc,
|
|
optOffsetX, optOffsetY
|
|
|
|
|
|
/* put temporarily disabled options here after optLast */
|
|
/*
|
|
optLamp,
|
|
*/
|
|
|
|
}
|
|
EOptionIndex;
|
|
|
|
typedef union
|
|
{
|
|
SANE_Word w;
|
|
SANE_Word *wa; /* word array */
|
|
SANE_String s;
|
|
}
|
|
TOptionValue;
|
|
|
|
|
|
typedef struct
|
|
{
|
|
SANE_Option_Descriptor aOptions[optLast];
|
|
TOptionValue aValues[optLast];
|
|
|
|
TScanParams ScanParams;
|
|
THWParams HWParams;
|
|
|
|
TDataPipe DataPipe;
|
|
int iLinesLeft;
|
|
|
|
SANE_Int *aGammaTableR; /* a 16-to-16 bit color lookup table */
|
|
SANE_Int *aGammaTableG; /* a 16-to-16 bit color lookup table */
|
|
SANE_Int *aGammaTableB; /* a 16-to-16 bit color lookup table */
|
|
|
|
int fScanning; /* TRUE if actively scanning */
|
|
int fCanceled;
|
|
}
|
|
TScanner;
|
|
|
|
|
|
/* linked list of SANE_Device structures */
|
|
typedef struct TDevListEntry
|
|
{
|
|
struct TDevListEntry *pNext;
|
|
SANE_Device dev;
|
|
char* devname;
|
|
}
|
|
TDevListEntry;
|
|
|
|
|
|
|
|
/* Device filename for USB access */
|
|
char usb_devfile[128];
|
|
|
|
static TDevListEntry *_pFirstSaneDev = 0;
|
|
static int iNumSaneDev = 0;
|
|
|
|
|
|
static const SANE_Device **_pSaneDevList = 0;
|
|
|
|
/* option constraints */
|
|
static const SANE_Range rangeGammaTable = {0, 65535, 1};
|
|
#ifdef SUPPORT_2400_DPI
|
|
static const SANE_Int setResolutions[] = {6, 75, 150, 300, 600, 1200, 2400};
|
|
#else
|
|
static const SANE_Int setResolutions[] = {5, 75, 150, 300, 600, 1200};
|
|
#endif
|
|
static const SANE_Range rangeXmm = {0, 220, 1};
|
|
static const SANE_Range rangeYmm = {0, 300, 1};
|
|
static const SANE_Range rangeXoffset = {0, 20, 1};
|
|
static const SANE_Range rangeYoffset = {0, 70, 1};
|
|
static const SANE_Int offsetX = 5;
|
|
static const SANE_Int offsetY = 52;
|
|
|
|
|
|
static void _InitOptions(TScanner *s)
|
|
{
|
|
int i, j;
|
|
SANE_Option_Descriptor *pDesc;
|
|
TOptionValue *pVal;
|
|
|
|
/* set a neutral gamma */
|
|
if( s->aGammaTableR == NULL ) /* Not yet allocated */
|
|
{
|
|
s->aGammaTableR = malloc( NUM_GAMMA_ENTRIES * sizeof( SANE_Int ) );
|
|
s->aGammaTableG = malloc( NUM_GAMMA_ENTRIES * sizeof( SANE_Int ) );
|
|
s->aGammaTableB = malloc( NUM_GAMMA_ENTRIES * sizeof( SANE_Int ) );
|
|
|
|
for (j = 0; j < NUM_GAMMA_ENTRIES; j++) {
|
|
s->aGammaTableR[j] = j;
|
|
s->aGammaTableG[j] = j;
|
|
s->aGammaTableB[j] = j;
|
|
}
|
|
}
|
|
|
|
for (i = optCount; i < optLast; i++) {
|
|
|
|
pDesc = &s->aOptions[i];
|
|
pVal = &s->aValues[i];
|
|
|
|
/* defaults */
|
|
pDesc->name = "";
|
|
pDesc->title = "";
|
|
pDesc->desc = "";
|
|
pDesc->type = SANE_TYPE_INT;
|
|
pDesc->unit = SANE_UNIT_NONE;
|
|
pDesc->size = sizeof(SANE_Word);
|
|
pDesc->constraint_type = SANE_CONSTRAINT_NONE;
|
|
pDesc->cap = 0;
|
|
|
|
switch (i) {
|
|
|
|
case optCount:
|
|
pDesc->title = SANE_TITLE_NUM_OPTIONS;
|
|
pDesc->desc = SANE_DESC_NUM_OPTIONS;
|
|
pDesc->cap = SANE_CAP_SOFT_DETECT;
|
|
pVal->w = (SANE_Word)optLast;
|
|
break;
|
|
|
|
case optGroupGeometry:
|
|
pDesc->title = "Geometry";
|
|
pDesc->type = SANE_TYPE_GROUP;
|
|
pDesc->size = 0;
|
|
break;
|
|
|
|
case optTLX:
|
|
pDesc->name = SANE_NAME_SCAN_TL_X;
|
|
pDesc->title = SANE_TITLE_SCAN_TL_X;
|
|
pDesc->desc = SANE_DESC_SCAN_TL_X;
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeXmm;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->w = rangeXmm.min + offsetX;
|
|
break;
|
|
|
|
case optTLY:
|
|
pDesc->name = SANE_NAME_SCAN_TL_Y;
|
|
pDesc->title = SANE_TITLE_SCAN_TL_Y;
|
|
pDesc->desc = SANE_DESC_SCAN_TL_Y;
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeYmm;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->w = rangeYmm.min + offsetY;
|
|
break;
|
|
|
|
case optBRX:
|
|
pDesc->name = SANE_NAME_SCAN_BR_X;
|
|
pDesc->title = SANE_TITLE_SCAN_BR_X;
|
|
pDesc->desc = SANE_DESC_SCAN_BR_X;
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeXmm;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->w = rangeXmm.max + offsetX;
|
|
break;
|
|
|
|
case optBRY:
|
|
pDesc->name = SANE_NAME_SCAN_BR_Y;
|
|
pDesc->title = SANE_TITLE_SCAN_BR_Y;
|
|
pDesc->desc = SANE_DESC_SCAN_BR_Y;
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeYmm;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->w = rangeYmm.max + offsetY;
|
|
break;
|
|
|
|
case optDPI:
|
|
pDesc->name = SANE_NAME_SCAN_RESOLUTION;
|
|
pDesc->title = SANE_TITLE_SCAN_RESOLUTION;
|
|
pDesc->desc = SANE_DESC_SCAN_RESOLUTION;
|
|
pDesc->unit = SANE_UNIT_DPI;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_WORD_LIST;
|
|
pDesc->constraint.word_list = setResolutions;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->w = setResolutions[1];
|
|
break;
|
|
|
|
case optGroupImage:
|
|
pDesc->title = SANE_I18N("Image");
|
|
pDesc->type = SANE_TYPE_GROUP;
|
|
pDesc->size = 0;
|
|
break;
|
|
|
|
case optGammaTableRed:
|
|
pDesc->name = SANE_NAME_GAMMA_VECTOR_R;
|
|
pDesc->title = SANE_TITLE_GAMMA_VECTOR_R;
|
|
pDesc->desc = SANE_DESC_GAMMA_VECTOR_R;
|
|
pDesc->size = NUM_GAMMA_ENTRIES * sizeof( SANE_Int );
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeGammaTable;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->wa = s->aGammaTableR;
|
|
break;
|
|
|
|
case optGammaTableGreen:
|
|
pDesc->name = SANE_NAME_GAMMA_VECTOR_G;
|
|
pDesc->title = SANE_TITLE_GAMMA_VECTOR_G;
|
|
pDesc->desc = SANE_DESC_GAMMA_VECTOR_G;
|
|
pDesc->size = NUM_GAMMA_ENTRIES * sizeof( SANE_Int );
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeGammaTable;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->wa = s->aGammaTableG;
|
|
break;
|
|
|
|
case optGammaTableBlue:
|
|
pDesc->name = SANE_NAME_GAMMA_VECTOR_B;
|
|
pDesc->title = SANE_TITLE_GAMMA_VECTOR_B;
|
|
pDesc->desc = SANE_DESC_GAMMA_VECTOR_B;
|
|
pDesc->size = NUM_GAMMA_ENTRIES * sizeof( SANE_Int );
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeGammaTable;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
pVal->wa = s->aGammaTableB;
|
|
break;
|
|
|
|
case optGroupMisc:
|
|
pDesc->title = SANE_I18N("Miscellaneous");
|
|
pDesc->type = SANE_TYPE_GROUP;
|
|
pDesc->size = 0;
|
|
break;
|
|
|
|
case optOffsetX:
|
|
pDesc->title = SANE_I18N("offset X");
|
|
pDesc->desc = SANE_I18N("Hardware internal X position of the scanning area.");
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeXoffset;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT;
|
|
pVal->w = offsetX;
|
|
break;
|
|
|
|
case optOffsetY:
|
|
pDesc->title = SANE_I18N("offset Y");
|
|
pDesc->desc = SANE_I18N("Hardware internal Y position of the scanning area.");
|
|
pDesc->unit = SANE_UNIT_MM;
|
|
pDesc->constraint_type = SANE_CONSTRAINT_RANGE;
|
|
pDesc->constraint.range = &rangeYoffset;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT;
|
|
pVal->w = offsetY;
|
|
break;
|
|
|
|
|
|
#if 0
|
|
case optLamp:
|
|
pDesc->name = "lamp";
|
|
pDesc->title = SANE_I18N("Lamp status");
|
|
pDesc->desc = SANE_I18N("Switches the lamp on or off.");
|
|
pDesc->type = SANE_TYPE_BOOL;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT;
|
|
/* switch the lamp on when starting for first the time */
|
|
pVal->w = SANE_TRUE;
|
|
break;
|
|
#endif
|
|
#if 0
|
|
case optCalibrate:
|
|
pDesc->name = "calibrate";
|
|
pDesc->title = SANE_I18N("Calibrate");
|
|
pDesc->desc = SANE_I18N("Calibrates for black and white level.");
|
|
pDesc->type = SANE_TYPE_BUTTON;
|
|
pDesc->cap = SANE_CAP_SOFT_SELECT;
|
|
pDesc->size = 0;
|
|
break;
|
|
#endif
|
|
default:
|
|
HP5400_DBG(DBG_ERR, "Uninitialised option %d\n", i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static int _ReportDevice(TScannerModel *pModel, const char *pszDeviceName)
|
|
{
|
|
TDevListEntry *pNew, *pDev;
|
|
|
|
HP5400_DBG(DBG_MSG, "hp5400: _ReportDevice '%s'\n", pszDeviceName);
|
|
|
|
pNew = malloc(sizeof(TDevListEntry));
|
|
if (!pNew) {
|
|
HP5400_DBG(DBG_ERR, "no mem\n");
|
|
return -1;
|
|
}
|
|
|
|
/* add new element to the end of the list */
|
|
if (_pFirstSaneDev == NULL) {
|
|
_pFirstSaneDev = pNew;
|
|
}
|
|
else {
|
|
for (pDev = _pFirstSaneDev; pDev->pNext; pDev = pDev->pNext) {
|
|
;
|
|
}
|
|
pDev->pNext = pNew;
|
|
}
|
|
|
|
/* fill in new element */
|
|
pNew->pNext = 0;
|
|
/* we use devname to avoid having to free a const
|
|
* pointer */
|
|
pNew->devname = (char*)strdup(pszDeviceName);
|
|
pNew->dev.name = pNew->devname;
|
|
pNew->dev.vendor = pModel->pszVendor;
|
|
pNew->dev.model = pModel->pszName;
|
|
pNew->dev.type = "flatbed scanner";
|
|
|
|
iNumSaneDev++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static SANE_Status
|
|
attach_one_device (SANE_String_Const devname)
|
|
{
|
|
const char * filename = (const char*) devname;
|
|
if (HP5400Detect (filename, _ReportDevice) < 0)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "attach_one_device: couldn't attach %s\n", devname);
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
HP5400_DBG (DBG_MSG, "attach_one_device: attached %s successfully\n", devname);
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
SANE_Status
|
|
sane_init (SANE_Int * piVersion, SANE_Auth_Callback pfnAuth)
|
|
{
|
|
FILE *conf_fp; /* Config file stream */
|
|
SANE_Char line[PATH_MAX];
|
|
SANE_Char *str = NULL;
|
|
SANE_String_Const proper_str;
|
|
int nline = 0;
|
|
|
|
/* prevent compiler from complaing about unused parameters */
|
|
pfnAuth = pfnAuth;
|
|
|
|
strcpy(usb_devfile, "/dev/usb/scanner0");
|
|
_pFirstSaneDev = 0;
|
|
iNumSaneDev = 0;
|
|
|
|
InitHp5400_internal();
|
|
|
|
|
|
DBG_INIT ();
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_init: SANE hp5400 backend version %d.%d-%d (from %s)\n",
|
|
SANE_CURRENT_MAJOR, V_MINOR, BUILD, PACKAGE_STRING);
|
|
|
|
sanei_usb_init ();
|
|
|
|
conf_fp = sanei_config_open (HP5400_CONFIG_FILE);
|
|
|
|
iNumSaneDev = 0;
|
|
|
|
if (conf_fp)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "Reading config file\n");
|
|
|
|
while (sanei_config_read (line, sizeof (line), conf_fp))
|
|
{
|
|
++nline;
|
|
|
|
if (str)
|
|
{
|
|
free (str);
|
|
}
|
|
|
|
proper_str = sanei_config_get_string (line, &str);
|
|
|
|
/* Discards white lines and comments */
|
|
if (!str || proper_str == line || str[0] == '#')
|
|
{
|
|
HP5400_DBG (DBG_MSG, "Discarding line %d\n", nline);
|
|
}
|
|
else
|
|
{
|
|
/* If line's not blank or a comment, then it's the device
|
|
* filename or a usb directive. */
|
|
HP5400_DBG (DBG_MSG, "Trying to attach %s\n", line);
|
|
sanei_usb_attach_matching_devices (line, attach_one_device);
|
|
}
|
|
} /* while */
|
|
fclose (conf_fp);
|
|
}
|
|
else
|
|
{
|
|
HP5400_DBG (DBG_ERR, "Unable to read config file \"%s\": %s\n",
|
|
HP5400_CONFIG_FILE, strerror (errno));
|
|
HP5400_DBG (DBG_MSG, "Using default built-in values\n");
|
|
attach_one_device (usb_devfile);
|
|
}
|
|
|
|
if (piVersion != NULL)
|
|
{
|
|
*piVersion = SANE_VERSION_CODE (SANE_CURRENT_MAJOR, V_MINOR, BUILD);
|
|
}
|
|
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
void
|
|
sane_exit (void)
|
|
{
|
|
TDevListEntry *pDev, *pNext;
|
|
HP5400_DBG (DBG_MSG, "sane_exit\n");
|
|
|
|
/* free device list memory */
|
|
if (_pSaneDevList)
|
|
{
|
|
for (pDev = _pFirstSaneDev; pDev; pDev = pNext)
|
|
{
|
|
pNext = pDev->pNext;
|
|
free (pDev->devname);
|
|
/* pDev->dev.name is the same pointer that pDev->devname */
|
|
free (pDev);
|
|
}
|
|
_pFirstSaneDev = 0;
|
|
free (_pSaneDevList);
|
|
_pSaneDevList = 0;
|
|
}
|
|
|
|
|
|
FreeHp5400_internal();
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only)
|
|
{
|
|
TDevListEntry *pDev;
|
|
int i;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_get_devices\n");
|
|
|
|
local_only = local_only;
|
|
|
|
if (_pSaneDevList)
|
|
{
|
|
free (_pSaneDevList);
|
|
}
|
|
|
|
_pSaneDevList = malloc (sizeof (*_pSaneDevList) * (iNumSaneDev + 1));
|
|
if (!_pSaneDevList)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "no mem\n");
|
|
return SANE_STATUS_NO_MEM;
|
|
}
|
|
i = 0;
|
|
for (pDev = _pFirstSaneDev; pDev; pDev = pDev->pNext)
|
|
{
|
|
_pSaneDevList[i++] = &pDev->dev;
|
|
}
|
|
_pSaneDevList[i++] = 0; /* last entry is 0 */
|
|
|
|
*device_list = _pSaneDevList;
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_open (SANE_String_Const name, SANE_Handle * h)
|
|
{
|
|
TScanner *s;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_open: %s\n", name);
|
|
|
|
/* check the name */
|
|
if (strlen (name) == 0)
|
|
{
|
|
/* default to first available device */
|
|
name = _pFirstSaneDev->dev.name;
|
|
}
|
|
|
|
s = malloc (sizeof (TScanner));
|
|
if (!s)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "malloc failed\n");
|
|
return SANE_STATUS_NO_MEM;
|
|
}
|
|
|
|
memset (s, 0, sizeof (TScanner)); /* Clear everything to zero */
|
|
if (HP5400Open (&s->HWParams, name) < 0)
|
|
{
|
|
/* is this OK ? */
|
|
HP5400_DBG (DBG_ERR, "HP5400Open failed\n");
|
|
free ((void *) s);
|
|
return SANE_STATUS_INVAL; /* is this OK? */
|
|
}
|
|
HP5400_DBG (DBG_MSG, "Handle=%d\n", s->HWParams.iXferHandle);
|
|
_InitOptions (s);
|
|
*h = s;
|
|
|
|
/* Turn on lamp by default at startup */
|
|
/* SetLamp(&s->HWParams, TRUE); */
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
void
|
|
sane_close (SANE_Handle h)
|
|
{
|
|
TScanner *s;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_close\n");
|
|
|
|
s = (TScanner *) h;
|
|
|
|
/* turn of scanner lamp */
|
|
SetLamp (&s->HWParams, FALSE);
|
|
|
|
/* close scanner */
|
|
HP5400Close (&s->HWParams);
|
|
|
|
/* free scanner object memory */
|
|
free ((void *) s);
|
|
}
|
|
|
|
|
|
const SANE_Option_Descriptor *
|
|
sane_get_option_descriptor (SANE_Handle h, SANE_Int n)
|
|
{
|
|
TScanner *s;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_get_option_descriptor %d\n", n);
|
|
|
|
if ((n < optCount) || (n >= optLast))
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
s = (TScanner *) h;
|
|
return &s->aOptions[n];
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_control_option (SANE_Handle h, SANE_Int n, SANE_Action Action,
|
|
void *pVal, SANE_Int * pInfo)
|
|
{
|
|
TScanner *s;
|
|
SANE_Int info;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_control_option: option %d, action %d\n", n, Action);
|
|
|
|
s = (TScanner *) h;
|
|
info = 0;
|
|
|
|
switch (Action)
|
|
{
|
|
case SANE_ACTION_GET_VALUE:
|
|
switch (n)
|
|
{
|
|
|
|
/* Get options of type SANE_Word */
|
|
case optBRX:
|
|
case optTLX:
|
|
*(SANE_Word *) pVal = s->aValues[n].w; /* Not needed anymore - s->aValues[optOffsetX].w; */
|
|
HP5400_DBG (DBG_MSG,
|
|
"sane_control_option: SANE_ACTION_GET_VALUE %d = %d\n", n,
|
|
*(SANE_Word *) pVal);
|
|
break;
|
|
|
|
case optBRY:
|
|
case optTLY:
|
|
*(SANE_Word *) pVal = s->aValues[n].w; /* Not needed anymore - - s->aValues[optOffsetY].w; */
|
|
HP5400_DBG (DBG_MSG,
|
|
"sane_control_option: SANE_ACTION_GET_VALUE %d = %d\n", n,
|
|
*(SANE_Word *) pVal);
|
|
break;
|
|
|
|
case optOffsetX:
|
|
case optOffsetY:
|
|
case optCount:
|
|
case optDPI:
|
|
HP5400_DBG (DBG_MSG,
|
|
"sane_control_option: SANE_ACTION_GET_VALUE %d = %d\n", n,
|
|
(int) s->aValues[n].w);
|
|
*(SANE_Word *) pVal = s->aValues[n].w;
|
|
break;
|
|
|
|
/* Get options of type SANE_Word array */
|
|
case optGammaTableRed:
|
|
case optGammaTableGreen:
|
|
case optGammaTableBlue:
|
|
HP5400_DBG (DBG_MSG, "Reading gamma table\n");
|
|
memcpy (pVal, s->aValues[n].wa, s->aOptions[n].size);
|
|
break;
|
|
|
|
#if 0
|
|
/* Get options of type SANE_Bool */
|
|
case optLamp:
|
|
GetLamp (&s->HWParams, &fLampIsOn);
|
|
*(SANE_Bool *) pVal = fLampIsOn;
|
|
break;
|
|
#endif
|
|
#if 0
|
|
case optCalibrate:
|
|
/* although this option has nothing to read,
|
|
it's added here to avoid a warning when running scanimage --help */
|
|
break;
|
|
#endif
|
|
default:
|
|
HP5400_DBG (DBG_MSG, "SANE_ACTION_GET_VALUE: Invalid option (%d)\n", n);
|
|
}
|
|
break;
|
|
|
|
|
|
case SANE_ACTION_SET_VALUE:
|
|
if (s->fScanning)
|
|
{
|
|
HP5400_DBG (DBG_ERR,
|
|
"sane_control_option: SANE_ACTION_SET_VALUE not allowed during scan\n");
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
switch (n)
|
|
{
|
|
|
|
case optCount:
|
|
return SANE_STATUS_INVAL;
|
|
break;
|
|
|
|
case optBRX:
|
|
case optTLX:
|
|
info |= SANE_INFO_RELOAD_PARAMS;
|
|
s->ScanParams.iLines = 0; /* Forget actual image settings */
|
|
s->aValues[n].w = *(SANE_Word *) pVal; /* Not needed anymore - + s->aValues[optOffsetX].w; */
|
|
break;
|
|
|
|
case optBRY:
|
|
case optTLY:
|
|
info |= SANE_INFO_RELOAD_PARAMS;
|
|
s->ScanParams.iLines = 0; /* Forget actual image settings */
|
|
s->aValues[n].w = *(SANE_Word *) pVal; /* Not needed anymore - + s->aValues[optOffsetY].w; */
|
|
break;
|
|
case optDPI:
|
|
info |= SANE_INFO_RELOAD_PARAMS;
|
|
s->ScanParams.iLines = 0; /* Forget actual image settings */
|
|
#ifdef SUPPORT_2400_DPI
|
|
(s->aValues[n].w) = *(SANE_Word *) pVal;
|
|
#else
|
|
(s->aValues[n].w) = min (1200, *(SANE_Word *) pVal);
|
|
#endif
|
|
break;
|
|
|
|
case optGammaTableRed:
|
|
case optGammaTableGreen:
|
|
case optGammaTableBlue:
|
|
HP5400_DBG (DBG_MSG, "Writing gamma table\n");
|
|
memcpy (s->aValues[n].wa, pVal, s->aOptions[n].size);
|
|
break;
|
|
/*
|
|
case optLamp:
|
|
fVal = *(SANE_Bool *)pVal;
|
|
HP5400_DBG(DBG_MSG, "lamp %s\n", fVal ? "on" : "off");
|
|
SetLamp(&s->HWParams, fVal);
|
|
break;
|
|
*/
|
|
#if 0
|
|
case optCalibrate:
|
|
/* SimpleCalib(&s->HWParams); */
|
|
break;
|
|
#endif
|
|
default:
|
|
HP5400_DBG (DBG_ERR, "SANE_ACTION_SET_VALUE: Invalid option (%d)\n", n);
|
|
}
|
|
if (pInfo != NULL)
|
|
{
|
|
*pInfo = info;
|
|
}
|
|
break;
|
|
|
|
case SANE_ACTION_SET_AUTO:
|
|
return SANE_STATUS_UNSUPPORTED;
|
|
|
|
|
|
default:
|
|
HP5400_DBG (DBG_ERR, "Invalid action (%d)\n", Action);
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
sane_get_parameters (SANE_Handle h, SANE_Parameters * p)
|
|
{
|
|
TScanner *s;
|
|
HP5400_DBG (DBG_MSG, "sane_get_parameters\n");
|
|
|
|
s = (TScanner *) h;
|
|
|
|
/* first do some checks */
|
|
if (s->aValues[optTLX].w >= s->aValues[optBRX].w)
|
|
{
|
|
HP5400_DBG (DBG_ERR, "TLX should be smaller than BRX\n");
|
|
return SANE_STATUS_INVAL; /* proper error code? */
|
|
}
|
|
if (s->aValues[optTLY].w >= s->aValues[optBRY].w)
|
|
{
|
|
HP5400_DBG (DBG_ERR, "TLY should be smaller than BRY\n");
|
|
return SANE_STATUS_INVAL; /* proper error code? */
|
|
}
|
|
|
|
/* return the data */
|
|
p->format = SANE_FRAME_RGB;
|
|
p->last_frame = SANE_TRUE;
|
|
|
|
p->depth = 8;
|
|
if (s->ScanParams.iLines) /* Initialised by doing a scan */
|
|
{
|
|
p->pixels_per_line = s->ScanParams.iBytesPerLine / 3;
|
|
p->lines = s->ScanParams.iLines;
|
|
p->bytes_per_line = s->ScanParams.iBytesPerLine;
|
|
}
|
|
else
|
|
{
|
|
p->lines = MM_TO_PIXEL (s->aValues[optBRY].w - s->aValues[optTLY].w,
|
|
s->aValues[optDPI].w);
|
|
p->pixels_per_line =
|
|
MM_TO_PIXEL (s->aValues[optBRX].w - s->aValues[optTLX].w,
|
|
s->aValues[optDPI].w);
|
|
p->bytes_per_line = p->pixels_per_line * 3;
|
|
}
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
#define BUFFER_READ_HEADER_SIZE 32
|
|
|
|
SANE_Status
|
|
sane_start (SANE_Handle h)
|
|
{
|
|
TScanner *s;
|
|
SANE_Parameters par;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_start\n");
|
|
|
|
s = (TScanner *) h;
|
|
|
|
if (sane_get_parameters (h, &par) != SANE_STATUS_GOOD)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "Invalid scan parameters (sane_get_parameters)\n");
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
s->iLinesLeft = par.lines;
|
|
|
|
/* fill in the scanparams using the option values */
|
|
s->ScanParams.iDpi = s->aValues[optDPI].w;
|
|
s->ScanParams.iLpi = s->aValues[optDPI].w;
|
|
|
|
/* Guessing here. 75dpi => 1, 2400dpi => 32 */
|
|
/* s->ScanParams.iColourOffset = s->aValues[optDPI].w / 75; */
|
|
/* now we don't need correction => corrected by scan request type ? */
|
|
s->ScanParams.iColourOffset = 0;
|
|
|
|
s->ScanParams.iTop =
|
|
MM_TO_PIXEL (s->aValues[optTLY].w + s->HWParams.iTopLeftY, HW_LPI);
|
|
s->ScanParams.iLeft =
|
|
MM_TO_PIXEL (s->aValues[optTLX].w + s->HWParams.iTopLeftX, HW_DPI);
|
|
|
|
/* Note: All measurements passed to the scanning routines must be in HW_LPI */
|
|
s->ScanParams.iWidth =
|
|
MM_TO_PIXEL (s->aValues[optBRX].w - s->aValues[optTLX].w, HW_LPI);
|
|
s->ScanParams.iHeight =
|
|
MM_TO_PIXEL (s->aValues[optBRY].w - s->aValues[optTLY].w, HW_LPI);
|
|
|
|
/* After the scanning, the iLines and iBytesPerLine will be filled in */
|
|
|
|
/* copy gamma table */
|
|
WriteGammaCalibTable (s->HWParams.iXferHandle, s->aGammaTableR,
|
|
s->aGammaTableG, s->aGammaTableB);
|
|
|
|
/* prepare the actual scan */
|
|
/* We say normal here. In future we should have a preview flag to set preview mode */
|
|
if (InitScan (SCAN_TYPE_NORMAL, &s->ScanParams, &s->HWParams) != 0)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "Invalid scan parameters (InitScan)\n");
|
|
return SANE_STATUS_INVAL;
|
|
}
|
|
|
|
/* for the moment no lines has been read */
|
|
s->ScanParams.iLinesRead = 0;
|
|
|
|
s->fScanning = TRUE;
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_read (SANE_Handle h, SANE_Byte * buf, SANE_Int maxlen, SANE_Int * len)
|
|
{
|
|
|
|
/* Read actual scan from the circular buffer */
|
|
/* Note: this is already color corrected, though some work still needs to be done
|
|
to deal with the colour offsetting */
|
|
TScanner *s;
|
|
char *buffer = (char*)buf;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_read: request %d bytes \n", maxlen);
|
|
|
|
s = (TScanner *) h;
|
|
|
|
/* nothing has been read for the moment */
|
|
*len = 0;
|
|
|
|
|
|
/* if we read all the lines return EOF */
|
|
if (s->ScanParams.iLinesRead == s->ScanParams.iLines)
|
|
{
|
|
/* FinishScan( &s->HWParams ); *** FinishScan called in sane_cancel */
|
|
HP5400_DBG (DBG_MSG, "sane_read: EOF\n");
|
|
return SANE_STATUS_EOF;
|
|
}
|
|
|
|
/* read as many lines the buffer may contain and while there are lines to be read */
|
|
while ((*len + s->ScanParams.iBytesPerLine <= maxlen)
|
|
&& (s->ScanParams.iLinesRead < s->ScanParams.iLines))
|
|
{
|
|
|
|
/* get one more line from the circular buffer */
|
|
CircBufferGetLine (s->HWParams.iXferHandle, &s->HWParams.pipe, buffer);
|
|
|
|
/* increment pointer, size and line number */
|
|
buffer += s->ScanParams.iBytesPerLine;
|
|
*len += s->ScanParams.iBytesPerLine;
|
|
s->ScanParams.iLinesRead++;
|
|
}
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_read: %d bytes read\n", *len);
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
void
|
|
sane_cancel (SANE_Handle h)
|
|
{
|
|
TScanner *s;
|
|
|
|
HP5400_DBG (DBG_MSG, "sane_cancel\n");
|
|
|
|
s = (TScanner *) h;
|
|
|
|
/* to be implemented more thoroughly */
|
|
|
|
/* Make sure the scanner head returns home */
|
|
FinishScan (&s->HWParams);
|
|
|
|
s->fCanceled = TRUE;
|
|
s->fScanning = FALSE;
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_set_io_mode (SANE_Handle h, SANE_Bool m)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "sane_set_io_mode %s\n", m ? "non-blocking" : "blocking");
|
|
|
|
/* prevent compiler from complaining about unused parameters */
|
|
h = h;
|
|
|
|
if (m)
|
|
{
|
|
return SANE_STATUS_UNSUPPORTED;
|
|
}
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
SANE_Status
|
|
sane_get_select_fd (SANE_Handle h, SANE_Int * fd)
|
|
{
|
|
HP5400_DBG (DBG_MSG, "sane_select_fd\n");
|
|
|
|
/* prevent compiler from complaining about unused parameters */
|
|
h = h;
|
|
fd = fd;
|
|
|
|
return SANE_STATUS_UNSUPPORTED;
|
|
}
|