kopia lustrzana https://gitlab.com/sane-project/backends
331 wiersze
8.7 KiB
C
331 wiersze
8.7 KiB
C
/*
|
|
Copyright (C) 2001 Bertrik Sikken (bertrik@zonnet.nl)
|
|
|
|
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.
|
|
|
|
$Id$
|
|
*/
|
|
|
|
/*
|
|
Provides a simple interface to read and write data from the scanner,
|
|
without any knowledge whether it's a parallel or USB scanner
|
|
*/
|
|
|
|
#include <stdio.h> /* printf */
|
|
#include <errno.h> /* better error reports */
|
|
#include <string.h> /* better error reports */
|
|
|
|
#include "niash_xfer.h"
|
|
|
|
#include "../include/sane/sanei_usb.h"
|
|
|
|
/* list of supported models */
|
|
STATIC TScannerModel ScannerModels[] = {
|
|
{"Hewlett-Packard", "ScanJet 3300C", 0x3F0, 0x205, eHp3300c}
|
|
,
|
|
{"Hewlett-Packard", "ScanJet 3400C", 0x3F0, 0x405, eHp3400c}
|
|
,
|
|
{"Hewlett-Packard", "ScanJet 4300C", 0x3F0, 0x305, eHp4300c}
|
|
,
|
|
{"Silitek Corp.", "HP ScanJet 4300c", 0x47b, 0x1002, eHp3400c}
|
|
,
|
|
{"Agfa", "Snapscan Touch", 0x6BD, 0x100, eAgfaTouch}
|
|
,
|
|
{"Trust", "Office Scanner USB 19200", 0x47b, 0x1000, eAgfaTouch}
|
|
,
|
|
/* last entry all zeros */
|
|
{0, 0, 0, 0, 0}
|
|
};
|
|
|
|
static TFnReportDevice *_pfnReportDevice;
|
|
static TScannerModel *_pModel;
|
|
|
|
/*
|
|
MatchUsbDevice
|
|
==============
|
|
Matches a given USB vendor and product id against a list of
|
|
supported scanners.
|
|
|
|
IN iVendor USB vendor ID
|
|
iProduct USB product ID
|
|
OUT *ppModel Pointer to TScannerModel structure
|
|
|
|
Returns TRUE if a matching USB scanner was found
|
|
*/
|
|
STATIC SANE_Bool
|
|
MatchUsbDevice (int iVendor, int iProduct, TScannerModel ** ppModel)
|
|
{
|
|
TScannerModel *pModels = ScannerModels;
|
|
|
|
DBG (DBG_MSG, "Matching USB device 0x%04X-0x%04X ... ", iVendor, iProduct);
|
|
while (pModels->pszName != NULL)
|
|
{
|
|
if ((pModels->iVendor == iVendor) && (pModels->iProduct == iProduct))
|
|
{
|
|
DBG (DBG_MSG, "found %s %s\n", pModels->pszVendor,
|
|
pModels->pszName);
|
|
*ppModel = pModels;
|
|
return SANE_TRUE;
|
|
}
|
|
/* next model to match */
|
|
pModels++;
|
|
}
|
|
DBG (DBG_MSG, "nothing found\n");
|
|
return SANE_FALSE;
|
|
}
|
|
|
|
/************************************************************************
|
|
Public functions for the SANE compilation
|
|
************************************************************************/
|
|
|
|
|
|
/* callback for sanei_usb_attach_matching_devices */
|
|
static SANE_Status
|
|
_AttachUsb (SANE_String_Const devname)
|
|
{
|
|
DBG (DBG_MSG, "_AttachUsb: found %s\n", devname);
|
|
|
|
_pfnReportDevice (_pModel, (const char *) devname);
|
|
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
|
|
/*
|
|
NiashXferInit
|
|
===============
|
|
Initialises all registered data transfer modules, which causes
|
|
them to report any devices found through the pfnReport callback.
|
|
|
|
IN pfnReport Function to call to report a transfer device
|
|
*/
|
|
static void
|
|
NiashXferInit (TFnReportDevice * pfnReport)
|
|
{
|
|
TScannerModel *pModels = ScannerModels;
|
|
|
|
sanei_usb_init ();
|
|
_pfnReportDevice = pfnReport;
|
|
|
|
/* loop over all scanner models */
|
|
while (pModels->pszName != NULL)
|
|
{
|
|
DBG (DBG_MSG, "Looking for %s...\n", pModels->pszName);
|
|
_pModel = pModels;
|
|
if (sanei_usb_find_devices ((SANE_Int) pModels->iVendor,
|
|
(SANE_Int) pModels->iProduct,
|
|
_AttachUsb) != SANE_STATUS_GOOD)
|
|
{
|
|
|
|
DBG (DBG_ERR, "Error invoking sanei_usb_find_devices");
|
|
break;
|
|
}
|
|
pModels++;
|
|
}
|
|
}
|
|
|
|
|
|
static int
|
|
NiashXferOpen (const char *pszName, EScannerModel * peModel)
|
|
{
|
|
SANE_Status status;
|
|
SANE_Word vendor, product;
|
|
int fd;
|
|
TScannerModel *pModel = 0;
|
|
|
|
DBG (DBG_MSG, "Trying to open %s...\n", pszName);
|
|
|
|
status = sanei_usb_open (pszName, &fd);
|
|
if (status != SANE_STATUS_GOOD)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
status = sanei_usb_get_vendor_product (fd, &vendor, &product);
|
|
if (status == SANE_STATUS_GOOD)
|
|
{
|
|
MatchUsbDevice (vendor, product, &pModel);
|
|
*peModel = pModel->eModel;
|
|
}
|
|
|
|
DBG (DBG_MSG, "handle = %d\n", (int) fd);
|
|
return fd;
|
|
}
|
|
|
|
|
|
static void
|
|
NiashXferClose (int iHandle)
|
|
{
|
|
/* close usb device */
|
|
if (iHandle != -1)
|
|
{
|
|
sanei_usb_close (iHandle);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
parusb_write_reg (int fd, unsigned char bReg, unsigned char bValue)
|
|
{
|
|
sanei_usb_control_msg (fd,
|
|
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_OUT,
|
|
0x0C, bReg, 0, 1, &bValue);
|
|
}
|
|
|
|
|
|
static void
|
|
parusb_read_reg (int fd, unsigned char bReg, unsigned char *pbValue)
|
|
{
|
|
sanei_usb_control_msg (fd,
|
|
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_IN,
|
|
0x0C, bReg, 0, 1, pbValue);
|
|
}
|
|
|
|
|
|
static void
|
|
NiashWriteReg (int iHandle, unsigned char bReg, unsigned char bData)
|
|
{
|
|
if (iHandle < 0)
|
|
{
|
|
DBG (DBG_MSG, "Invalid handle %d\n", iHandle);
|
|
return;
|
|
}
|
|
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, EPP_ADDR, bReg);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, EPP_DATA_WRITE, bData);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
}
|
|
|
|
|
|
static void
|
|
NiashReadReg (int iHandle, unsigned char bReg, unsigned char *pbData)
|
|
{
|
|
if (iHandle < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, EPP_ADDR, bReg);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x34);
|
|
parusb_read_reg (iHandle, EPP_DATA_READ, pbData);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
}
|
|
|
|
|
|
static void
|
|
NiashWriteBulk (int iHandle, unsigned char *pabBuf, int iSize)
|
|
{
|
|
/* byte abSetup[8] = {0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
|
HP3400 probably needs 0x01, 0x01 */
|
|
SANE_Byte abSetup[8] = { 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
|
size_t size;
|
|
|
|
if (iHandle < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* select scanner register 0x24 */
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, EPP_ADDR, 0x24);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
|
|
/* tell scanner that a bulk transfer follows */
|
|
abSetup[4] = (iSize) & 0xFF;
|
|
abSetup[5] = (iSize >> 8) & 0xFF;
|
|
sanei_usb_control_msg (iHandle,
|
|
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_OUT,
|
|
0x04, USB_SETUP, 0, 8, abSetup);
|
|
|
|
/* do the bulk write */
|
|
size = iSize;
|
|
if (sanei_usb_write_bulk (iHandle, pabBuf, &size) != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_ERR, "ERROR: Bulk write failed\n");
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
NiashReadBulk (int iHandle, unsigned char *pabBuf, int iSize)
|
|
{
|
|
SANE_Byte abSetup[8] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
|
size_t size;
|
|
|
|
if (iHandle < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* select scanner register 0x24 */
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, EPP_ADDR, 0x24);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
|
|
/* tell scanner that a bulk transfer follows */
|
|
abSetup[4] = (iSize) & 0xFF;
|
|
abSetup[5] = (iSize >> 8) & 0xFF;
|
|
sanei_usb_control_msg (iHandle,
|
|
USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_OUT,
|
|
0x04, USB_SETUP, 0, 8, abSetup);
|
|
|
|
/* do the bulk read */
|
|
size = iSize;
|
|
if (sanei_usb_read_bulk (iHandle, pabBuf, &size) != SANE_STATUS_GOOD)
|
|
{
|
|
DBG (DBG_ERR, "ERROR: Bulk read failed\n");
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
NiashWakeup (int iHandle)
|
|
{
|
|
unsigned char abMagic[] = { 0xA0, 0xA8, 0x50, 0x58, 0x90, 0x98, 0xC0, 0xC8,
|
|
0x90, 0x98, 0xE0, 0xE8
|
|
};
|
|
int i;
|
|
|
|
if (iHandle < 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* write magic startup sequence */
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
for (i = 0; i < (int) sizeof (abMagic); i++)
|
|
{
|
|
parusb_write_reg (iHandle, SPP_DATA, abMagic[i]);
|
|
}
|
|
|
|
/* write 0x04 to scanner register 0x00 the hard way */
|
|
parusb_write_reg (iHandle, SPP_DATA, 0x00);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x15);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x1D);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x15);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
|
|
parusb_write_reg (iHandle, SPP_DATA, 0x04);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x15);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x17);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x15);
|
|
parusb_write_reg (iHandle, SPP_CONTROL, 0x14);
|
|
}
|