kopia lustrzana https://gitlab.com/sane-project/backends
362 wiersze
8.9 KiB
C
362 wiersze
8.9 KiB
C
/** @file plustek-pp_wrapper.c
|
|
* @brief The interface to the parport driver-code and the kernel module.
|
|
*
|
|
* Based on sources acquired from Plustek Inc.<br>
|
|
* Copyright (C) 2001-2004 Gerhard Jaeger <gerhard@gjaeger.de>
|
|
*
|
|
* History:
|
|
* - 0.40 - initial version
|
|
* - 0.41 - added _PTDRV_ADJUST call
|
|
* - 0.42 - added setmap function
|
|
* - fixed the stopscan problem, that causes a crash in the kernel module
|
|
* - 0.43 - added initialized setting
|
|
* - cleanup
|
|
* .
|
|
* <hr>
|
|
* 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.
|
|
* <hr>
|
|
*/
|
|
|
|
/******************* wrapper functions for parport device ********************/
|
|
|
|
#ifndef _BACKEND_ENABLED
|
|
|
|
static int PtDrvInit( char *dev_name, unsigned short model_override )
|
|
{
|
|
_VAR_NOT_USED( dev_name );
|
|
_VAR_NOT_USED( model_override );
|
|
DBG( _DBG_ERROR, "Backend does not support direct I/O!\n" );
|
|
return -1;
|
|
}
|
|
|
|
static int PtDrvShutdown( void )
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int PtDrvOpen( void )
|
|
{
|
|
DBG( _DBG_ERROR, "Backend does not support direct I/O!\n" );
|
|
return -1;
|
|
}
|
|
|
|
static int PtDrvClose( void )
|
|
{
|
|
DBG( _DBG_ERROR, "Backend does not support direct I/O!\n" );
|
|
return 0;
|
|
}
|
|
|
|
static int PtDrvIoctl( unsigned int cmd, void *arg )
|
|
{
|
|
DBG( _DBG_ERROR, "Backend does not support direct I/O!\n" );
|
|
_VAR_NOT_USED( cmd );
|
|
if( arg == NULL )
|
|
return -2;
|
|
return -1;
|
|
}
|
|
|
|
static int PtDrvRead( unsigned char *buffer, int count )
|
|
{
|
|
DBG( _DBG_ERROR, "Backend does not support direct I/O!\n" );
|
|
_VAR_NOT_USED( count );
|
|
if( buffer == NULL )
|
|
return -2;
|
|
return -1;
|
|
}
|
|
#endif /* _BACKEND_ENABLED */
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_open( const char *dev_name, void *misc )
|
|
{
|
|
int result;
|
|
int handle;
|
|
CompatAdjDef compatAdj;
|
|
PPAdjDef adj;
|
|
unsigned short version = _PTDRV_IOCTL_VERSION;
|
|
Plustek_Device *dev = (Plustek_Device *)misc;
|
|
|
|
if( dev->adj.direct_io ) {
|
|
result = PtDrvInit( dev_name, dev->adj.mov );
|
|
if( 0 != result ) {
|
|
DBG( _DBG_ERROR, "open: PtDrvInit failed: %d\n", result );
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if( dev->adj.direct_io )
|
|
handle = PtDrvOpen();
|
|
else
|
|
handle = open( dev_name, O_RDONLY );
|
|
|
|
if ( handle < 0 ) {
|
|
DBG( _DBG_ERROR, "open: can't open %s as a device\n", dev_name );
|
|
return handle;
|
|
}
|
|
|
|
if( dev->adj.direct_io )
|
|
result = PtDrvIoctl( _PTDRV_OPEN_DEVICE, &version );
|
|
else
|
|
result = ioctl( handle, _PTDRV_OPEN_DEVICE, &version );
|
|
|
|
if( result < 0 ) {
|
|
|
|
if( -9019 == result ) {
|
|
|
|
DBG( _DBG_INFO, "Version 0x%04x not supported, trying "
|
|
"compatibility version 0x%04x\n",
|
|
_PTDRV_IOCTL_VERSION, _PTDRV_COMPAT_IOCTL_VERSION);
|
|
|
|
version = _PTDRV_COMPAT_IOCTL_VERSION;
|
|
|
|
if( dev->adj.direct_io )
|
|
result = PtDrvIoctl( _PTDRV_OPEN_DEVICE, &version );
|
|
else
|
|
result = ioctl( handle, _PTDRV_OPEN_DEVICE, &version );
|
|
|
|
if( result < 0 ) {
|
|
|
|
if( dev->adj.direct_io )
|
|
PtDrvClose();
|
|
else
|
|
close( dev->fd );
|
|
|
|
DBG( _DBG_ERROR,
|
|
"ioctl PT_DRV_OPEN_DEVICE failed(%d)\n", result );
|
|
|
|
if( -9019 == result ) {
|
|
DBG( _DBG_ERROR,
|
|
"Version problem, please recompile driver!\n" );
|
|
}
|
|
} else {
|
|
|
|
DBG( _DBG_INFO, "Using compatibility version\n" );
|
|
|
|
compatAdj.lampOff = dev->adj.lampOff;
|
|
compatAdj.lampOffOnEnd = dev->adj.lampOffOnEnd;
|
|
compatAdj.warmup = dev->adj.warmup;
|
|
|
|
memcpy( &compatAdj.pos, &dev->adj.pos, sizeof(OffsDef));
|
|
memcpy( &compatAdj.neg, &dev->adj.neg, sizeof(OffsDef));
|
|
memcpy( &compatAdj.tpa, &dev->adj.tpa, sizeof(OffsDef));
|
|
|
|
if( dev->adj.direct_io )
|
|
PtDrvIoctl( _PTDRV_ADJUST, &compatAdj );
|
|
else
|
|
ioctl( handle, _PTDRV_ADJUST, &compatAdj );
|
|
return handle;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
memset( &adj, 0, sizeof(PPAdjDef));
|
|
|
|
adj.lampOff = dev->adj.lampOff;
|
|
adj.lampOffOnEnd = dev->adj.lampOffOnEnd;
|
|
adj.warmup = dev->adj.warmup;
|
|
|
|
memcpy( &adj.pos, &dev->adj.pos, sizeof(OffsDef));
|
|
memcpy( &adj.neg, &dev->adj.neg, sizeof(OffsDef));
|
|
memcpy( &adj.tpa, &dev->adj.tpa, sizeof(OffsDef));
|
|
|
|
adj.rgamma = dev->adj.rgamma;
|
|
adj.ggamma = dev->adj.ggamma;
|
|
adj.bgamma = dev->adj.bgamma;
|
|
adj.graygamma = dev->adj.graygamma;
|
|
|
|
if( dev->adj.direct_io )
|
|
PtDrvIoctl( _PTDRV_ADJUST, &adj );
|
|
else
|
|
ioctl( handle, _PTDRV_ADJUST, &adj );
|
|
|
|
dev->initialized = SANE_TRUE;
|
|
return handle;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_close( Plustek_Device *dev )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvClose();
|
|
else
|
|
return close( dev->fd );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_getCaps( Plustek_Device *dev )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_GET_CAPABILITIES, &dev->caps );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_GET_CAPABILITIES, &dev->caps );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_getLensInfo( Plustek_Device *dev, pLensInfo lens )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_GET_LENSINFO, lens );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_GET_LENSINFO, lens );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_getCropInfo( Plustek_Device *dev, pCropInfo crop )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_GET_CROPINFO, crop );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_GET_CROPINFO, crop );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_putImgInfo( Plustek_Device *dev, pImgDef img )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_PUT_IMAGEINFO, img );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_PUT_IMAGEINFO, img );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_setScanEnv( Plustek_Device *dev, pScanInfo sinfo )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_SET_ENV, sinfo );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_SET_ENV, sinfo );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_startScan( Plustek_Device *dev, pStartScan start )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvIoctl( _PTDRV_START_SCAN, start );
|
|
else
|
|
return ioctl( dev->fd, _PTDRV_START_SCAN, start );
|
|
}
|
|
|
|
/** function to send a gamma table to the kernel module. As the default table
|
|
* entry is 16-bit, but the maps are 8-bit, we have to copy the values...
|
|
*/
|
|
static int ppDev_setMap( Plustek_Device *dev, SANE_Word *map,
|
|
SANE_Word length, SANE_Word channel )
|
|
{
|
|
SANE_Byte *buf;
|
|
SANE_Word i;
|
|
MapDef m;
|
|
|
|
m.len = length;
|
|
m.map_id = channel;
|
|
|
|
m.map = (void *)map;
|
|
|
|
DBG(_DBG_INFO,"Setting map[%u] at 0x%08lx\n", channel, (unsigned long)map);
|
|
|
|
buf = (SANE_Byte*)malloc( m.len );
|
|
|
|
if( !buf )
|
|
return _E_ALLOC;
|
|
|
|
for( i = 0; i < m.len; i++ ) {
|
|
buf[i] = (SANE_Byte)map[i];
|
|
|
|
if( map[i] > 0xFF )
|
|
buf[i] = 0xFF;
|
|
}
|
|
|
|
m.map = buf;
|
|
|
|
if( dev->adj.direct_io )
|
|
PtDrvIoctl( _PTDRV_SETMAP, &m );
|
|
else
|
|
ioctl( dev->fd, _PTDRV_SETMAP, &m );
|
|
|
|
/* we ignore the return values */
|
|
free( buf );
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_stopScan( Plustek_Device *dev, short *mode )
|
|
{
|
|
int retval, tmp;
|
|
|
|
/* save this one... */
|
|
tmp = *mode;
|
|
|
|
if( dev->adj.direct_io )
|
|
retval = PtDrvIoctl( _PTDRV_STOP_SCAN, mode );
|
|
else
|
|
retval = ioctl( dev->fd, _PTDRV_STOP_SCAN, mode );
|
|
|
|
/* ... and use it here */
|
|
if( 0 == tmp ) {
|
|
if( dev->adj.direct_io )
|
|
PtDrvIoctl( _PTDRV_CLOSE_DEVICE, 0 );
|
|
else
|
|
ioctl( dev->fd, _PTDRV_CLOSE_DEVICE, 0);
|
|
}else
|
|
sleep( 1 );
|
|
|
|
return retval;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static int ppDev_readImage( Plustek_Device *dev,
|
|
SANE_Byte *buf, unsigned long data_length )
|
|
{
|
|
if( dev->adj.direct_io )
|
|
return PtDrvRead( buf, data_length );
|
|
else
|
|
return read( dev->fd, buf, data_length );
|
|
}
|
|
|
|
/* END PLUSTEK-PP_WRAPPER.C .................................................*/
|