kopia lustrzana https://gitlab.com/sane-project/backends
520 wiersze
15 KiB
C
520 wiersze
15 KiB
C
/* @file u12-motor.c
|
|
* @brief all functions for motor control
|
|
*
|
|
* based on sources acquired from Plustek Inc.
|
|
* Copyright (C) 2003-2004 Gerhard Jaeger <gerhard@gjaeger.de>
|
|
*
|
|
* History:
|
|
* - 0.01 - initial version
|
|
* - 0.02 - no changes
|
|
* .
|
|
* <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>
|
|
*/
|
|
|
|
/*************************** some definitons *********************************/
|
|
|
|
#define _BACKSTEPS 120
|
|
#define _FORWARDSTEPS 120
|
|
|
|
/**************************** local vars *************************************/
|
|
|
|
static TimerDef u12motor_Timer;
|
|
|
|
|
|
/*************************** local functions *********************************/
|
|
|
|
/**
|
|
*/
|
|
static void u12motor_DownloadNullScanStates( U12_Device *dev )
|
|
{
|
|
memset( dev->scanStates, 0, _SCANSTATE_BYTES );
|
|
u12io_DownloadScanStates( dev );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static void u12motor_Force16Steps( U12_Device *dev, int dir )
|
|
{
|
|
u_long dw;
|
|
|
|
if( dir == _DIR_FW )
|
|
u12io_DataToRegister( dev, REG_MOTOR0CONTROL, _FORWARD_MOTOR );
|
|
else if( dir == _DIR_BW )
|
|
u12io_DataToRegister( dev, REG_MOTOR0CONTROL, _BACKWARD_MOTOR );
|
|
|
|
for( dw = 16; dw; dw-- ) {
|
|
u12io_RegisterToScanner( dev, REG_FORCESTEP );
|
|
_DODELAY( 10 );
|
|
}
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static void u12motor_ModuleFreeRun( U12_Device *dev, u_long steps )
|
|
{
|
|
SANE_Byte rb[6];
|
|
|
|
rb[0] = REG_MOTORFREERUNCOUNT1; rb[1] = _HIBYTE(steps);
|
|
rb[2] = REG_MOTORFREERUNCOUNT0; rb[3] = _LOBYTE(steps);
|
|
rb[4] = REG_MOTORFREERUNTRIGGER; rb[5] = 0;
|
|
|
|
u12io_DataToRegs( dev, rb, 3 );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static SANE_Status u12motor_PositionYProc( U12_Device *dev, u_long steps )
|
|
{
|
|
TimerDef timer;
|
|
|
|
DBG( _DBG_INFO, "u12motor_PositionYProc()\n" );
|
|
u12io_StartTimer( &timer, _SECOND * 5 );
|
|
|
|
u12io_ResetFifoLen();
|
|
while(!(u12io_GetScanState( dev ) & _SCANSTATE_STOP) &&
|
|
(!u12io_CheckTimer( &timer )));
|
|
_DODELAY( 12 );
|
|
u12motor_ModuleFreeRun( dev, steps );
|
|
_DODELAY( 15 );
|
|
|
|
u12io_StartTimer( &timer, _SECOND * 30 );
|
|
do {
|
|
if( !(u12io_GetExtendedStatus( dev ) & _STILL_FREE_RUNNING)) {
|
|
break;
|
|
}
|
|
|
|
if( u12io_IsEscPressed()) {
|
|
DBG( _DBG_INFO, "* CANCEL detected!\n" );
|
|
return SANE_STATUS_CANCELLED;
|
|
}
|
|
|
|
} while( !u12io_CheckTimer( &timer ));
|
|
DBG( _DBG_INFO, "u12motor_PositionYProc() - done\n" );
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/** initialize this module and setup the correct function pointer according
|
|
* to the ASIC
|
|
*/
|
|
static void u12motor_PositionModuleToHome( U12_Device *dev )
|
|
{
|
|
SANE_Byte rb[50];
|
|
SANE_Byte save, saveModel;
|
|
int c = 0;
|
|
|
|
DBG( _DBG_INFO, "u12motor_PositionModuleToHome()\n" );
|
|
saveModel = dev->regs.RD_ModelControl;
|
|
|
|
dev->scan.refreshState = SANE_FALSE;
|
|
u12motor_DownloadNullScanStates( dev );
|
|
|
|
_DODELAY( 125 );
|
|
save = dev->shade.intermediate;
|
|
|
|
dev->shade.intermediate = _ScanMode_AverageOut;
|
|
u12hw_InitAsic( dev, SANE_FALSE );
|
|
dev->shade.intermediate = save;
|
|
|
|
_SET_REG( rb, c, REG_MODECONTROL, _ModeScan );
|
|
_SET_REG( rb, c, REG_RESETMTSC, 0 );
|
|
_SET_REG( rb, c, REG_SCANCONTROL1, 0 );
|
|
_SET_REG( rb, c, REG_MODELCONTROL, (dev->ModelCtrl | _MODEL_DPI300));
|
|
_SET_REG( rb, c, REG_LINECONTROL, 80 );
|
|
_SET_REG( rb, c, REG_XSTEPTIME, dev->XStepBack );
|
|
_SET_REG( rb, c, REG_MOTORDRVTYPE, dev->MotorPower );
|
|
_SET_REG( rb, c, REG_MOTOR0CONTROL, (_MotorHHomeStop | _MotorOn |
|
|
_MotorHQuarterStep | _MotorPowerEnable));
|
|
_SET_REG( rb, c, REG_STEPCONTROL, (_MOTOR0_SCANSTATE | _MOTOR_FREERUN));
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
memset( dev->scanStates, 0x88, _SCANSTATE_BYTES );
|
|
u12io_DownloadScanStates( dev );
|
|
|
|
u12io_RegisterToScanner( dev, REG_REFRESHSCANSTATE );
|
|
dev->regs.RD_ModelControl = saveModel;
|
|
}
|
|
|
|
/** function to bring the sensor back home
|
|
*/
|
|
static void u12motor_ToHomePosition( U12_Device *dev, SANE_Bool wait )
|
|
{
|
|
TimerDef timer;
|
|
|
|
DBG( _DBG_INFO, "Waiting for Sensor to be back in position\n" );
|
|
if( !(u12io_DataFromRegister( dev, REG_STATUS ) & _FLAG_PAPER)) {
|
|
|
|
u12motor_PositionModuleToHome( dev );
|
|
|
|
if( wait ) {
|
|
u12io_StartTimer( &timer, _SECOND * 20);
|
|
do {
|
|
if( u12io_DataFromRegister( dev, REG_STATUS ) & _FLAG_PAPER)
|
|
break;
|
|
} while( !u12io_CheckTimer( &timer ));
|
|
}
|
|
}
|
|
DBG( _DBG_INFO, "- done !\n" );
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static SANE_Status u12motor_BackToHomeSensor( U12_Device *dev )
|
|
{
|
|
SANE_Byte rb[20];
|
|
int c;
|
|
TimerDef timer;
|
|
|
|
DBG( _DBG_INFO, "u12Motor_BackToHomeSensor()\n" );
|
|
|
|
c = 0;
|
|
_SET_REG( rb, c, REG_STEPCONTROL, _MOTOR0_SCANSTATE );
|
|
_SET_REG( rb, c, REG_MODECONTROL, _ModeScan );
|
|
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
u12motor_Force16Steps( dev, _DIR_NONE );
|
|
|
|
/* stepping every state */
|
|
memset( dev->scanStates, 0x88, _SCANSTATE_BYTES );
|
|
u12io_DownloadScanStates( dev );
|
|
_DODELAY(50);
|
|
|
|
u12io_StartTimer( &timer, _SECOND * 2 );
|
|
|
|
u12io_ResetFifoLen();
|
|
while(!(u12io_GetScanState( dev ) & _SCANSTATE_STOP) &&
|
|
!u12io_CheckTimer( &timer )) {
|
|
if( u12io_IsEscPressed()) {
|
|
DBG( _DBG_INFO, "* CANCEL detected!\n" );
|
|
return SANE_STATUS_CANCELLED;
|
|
}
|
|
}
|
|
|
|
u12motor_Force16Steps( dev, _DIR_BW );
|
|
dev->regs.RD_ModeControl = _ModeScan;
|
|
|
|
c = 0;
|
|
|
|
if(!(dev->DataInf.dwScanFlag & _SCANDEF_TPA)) {
|
|
_SET_REG( rb, c, REG_LINECONTROL, _LOBYTE(dev->shade.wExposure));
|
|
_SET_REG( rb, c, REG_XSTEPTIME, _LOBYTE(dev->shade.wXStep));
|
|
} else {
|
|
_SET_REG( rb, c, REG_LINECONTROL, _DEFAULT_LINESCANTIME );
|
|
_SET_REG( rb, c, REG_XSTEPTIME, 6 );
|
|
}
|
|
|
|
_SET_REG( rb, c, REG_STEPCONTROL, (_MOTOR_FREERUN | _MOTOR0_SCANSTATE));
|
|
_SET_REG( rb, c, REG_MOTOR0CONTROL,
|
|
(_MotorHQuarterStep | _MotorOn | _MotorDirBackward |
|
|
_MotorPowerEnable | _MotorHHomeStop));
|
|
_SET_REG( rb, c, REG_REFRESHSCANSTATE, 0);
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
u12io_StartTimer( &timer, _SECOND * 5 );
|
|
do {
|
|
if( u12io_DataFromRegister( dev, REG_STATUS ) & _FLAG_PAPER )
|
|
break;
|
|
|
|
if( u12io_IsEscPressed()) {
|
|
DBG( _DBG_INFO, "* CANCEL detected!\n" );
|
|
return SANE_STATUS_CANCELLED;
|
|
}
|
|
|
|
_DODELAY( 55 );
|
|
|
|
} while( !u12io_CheckTimer( &timer ));
|
|
|
|
c = 0;
|
|
_SET_REG( rb, c, REG_LINECONTROL, dev->regs.RD_LineControl);
|
|
_SET_REG( rb, c, REG_XSTEPTIME, dev->regs.RD_XStepTime);
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
DBG( _DBG_INFO, "* LineCtrl=0x%02x, XStepTime=0x%02x\n",
|
|
dev->regs.RD_LineControl, dev->regs.RD_XStepTime );
|
|
|
|
u12motor_DownloadNullScanStates( dev );
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static SANE_Status u12motor_ModuleToHome( U12_Device *dev )
|
|
{
|
|
SANE_Status res;
|
|
|
|
DBG( _DBG_INFO, "u12motor_ModuleToHome()\n" );
|
|
if(!(u12io_DataFromRegister( dev, REG_STATUS ) & _FLAG_PAPER)) {
|
|
|
|
u12io_DataToRegister( dev, REG_MOTOR0CONTROL,
|
|
(SANE_Byte)(dev->regs.RD_Motor0Control|_MotorDirForward));
|
|
|
|
res = u12motor_PositionYProc( dev, 40 );
|
|
if( SANE_STATUS_GOOD != res )
|
|
return res;
|
|
|
|
res = u12motor_BackToHomeSensor( dev );
|
|
if( SANE_STATUS_GOOD != res )
|
|
return res;
|
|
|
|
_DODELAY( 250 );
|
|
}
|
|
DBG( _DBG_INFO, "* done.\n" );
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static SANE_Status u12motor_WaitForPositionY( U12_Device *dev )
|
|
{
|
|
SANE_Byte rb[20];
|
|
SANE_Status res;
|
|
SANE_Byte bXStep;
|
|
int c;
|
|
u_long dwBeginY;
|
|
|
|
c = 0;
|
|
dwBeginY = (u_long)dev->DataInf.crImage.y * 4 + dev->scan.dwScanOrigin;
|
|
|
|
if( dev->DataInf.wPhyDataType <= COLOR_256GRAY ) {
|
|
if( dev->f0_8_16 )
|
|
dwBeginY += 16;
|
|
else
|
|
dwBeginY += 8;
|
|
}
|
|
|
|
bXStep = (SANE_Byte)((dev->DataInf.wPhyDataType <= COLOR_256GRAY) ?
|
|
dev->XStepMono : dev->XStepColor);
|
|
|
|
if( dev->shade.intermediate & _ScanMode_AverageOut )
|
|
bXStep = 8;
|
|
|
|
u12motor_Force16Steps( dev, _DIR_NONE );
|
|
dwBeginY -= 16;
|
|
|
|
if( dwBeginY > (_RFT_SCANNING_ORG + _YOFFSET) &&
|
|
bXStep < dev->regs.RD_XStepTime ) {
|
|
|
|
u12io_DataToRegister( dev, REG_MOTORDRVTYPE, dev->MotorPower );
|
|
_DODELAY( 12 );
|
|
u12io_DataToRegister( dev, REG_XSTEPTIME, bXStep);
|
|
u12io_DataToRegister( dev, REG_EXTENDEDXSTEP, 0 );
|
|
u12io_DataToRegister( dev, REG_SCANCONTROL1,
|
|
(SANE_Byte)(dev->regs.RD_ScanControl1 & ~_MFRC_RUNSCANSTATE));
|
|
res = u12motor_PositionYProc( dev, dwBeginY - 64 );
|
|
if( SANE_STATUS_GOOD != res )
|
|
return res;
|
|
dwBeginY = 64;
|
|
} else {
|
|
_SET_REG(rb, c, REG_SCANCONTROL1, dev->regs.RD_ScanControl1 );
|
|
}
|
|
|
|
_SET_REG( rb, c, REG_FIFOFULLEN0, _LOBYTE(dev->regs.RD_BufFullSize));
|
|
_SET_REG( rb, c, REG_FIFOFULLEN1, _HIBYTE(dev->regs.RD_BufFullSize));
|
|
_SET_REG( rb, c, REG_FIFOFULLEN2, _LOBYTE(_HIWORD(dev->regs.RD_BufFullSize)));
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
u12io_DataToRegister( dev, REG_MOTORDRVTYPE, dev->regs.RD_MotorDriverType);
|
|
_DODELAY( 12 );
|
|
|
|
if(!dev->f2003 || (dev->shade.intermediate & _ScanMode_AverageOut) ||
|
|
( dev->DataInf.xyAppDpi.y <= 75 &&
|
|
dev->DataInf.wPhyDataType <= COLOR_256GRAY)) {
|
|
u12io_DataToRegister( dev, REG_MOTORDRVTYPE,
|
|
(SANE_Byte)(dev->MotorPower & (_MOTORR_MASK | _MOTORR_STRONG)));
|
|
} else {
|
|
u12io_DataToRegister( dev, REG_MOTORDRVTYPE,
|
|
dev->regs.RD_MotorDriverType );
|
|
}
|
|
|
|
c = 0;
|
|
_SET_REG( rb, c, REG_XSTEPTIME, dev->regs.RD_XStepTime );
|
|
_SET_REG( rb, c, REG_EXTENDEDXSTEP, dev->regs.RD_ExtXStepTime );
|
|
_SET_REG( rb, c, REG_SCANCONTROL1,
|
|
(SANE_Byte)(dev->regs.RD_ScanControl1 & ~_MFRC_RUNSCANSTATE));
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
if( dev->DataInf.dwScanFlag & _SCANDEF_PREVIEW ) {
|
|
|
|
TimerDef timer;
|
|
|
|
u12motor_ModuleFreeRun( dev, dwBeginY );
|
|
_DODELAY( 15 );
|
|
|
|
u12io_StartTimer( &timer, (_SECOND * 20));
|
|
|
|
while(( u12io_GetExtendedStatus( dev ) & _STILL_FREE_RUNNING) &&
|
|
!u12io_CheckTimer(&timer));
|
|
u12io_DataToRegister( dev, REG_MODECONTROL, _ModeScan );
|
|
} else {
|
|
u12motor_PositionYProc( dev, dwBeginY );
|
|
u12io_RegisterToScanner( dev, REG_REFRESHSCANSTATE );
|
|
}
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static void u12motor_ForceToLeaveHomePos( U12_Device *dev )
|
|
{
|
|
SANE_Byte rb[4];
|
|
TimerDef timer;
|
|
|
|
DBG( _DBG_INFO, "u12motor_ForceToLeaveHomePos()\n" );
|
|
rb[0] = REG_STEPCONTROL;
|
|
rb[1] = _MOTOR0_ONESTEP;
|
|
rb[2] = REG_MOTOR0CONTROL;
|
|
rb[3] = _FORWARD_MOTOR;
|
|
u12io_DataToRegs( dev, rb, 2 );
|
|
|
|
u12io_StartTimer( &timer, _SECOND );
|
|
|
|
do {
|
|
if( !(u12io_DataFromRegister( dev, REG_STATUS ) & _FLAG_PAPER))
|
|
break;
|
|
|
|
u12io_RegisterToScanner( dev, REG_FORCESTEP );
|
|
_DODELAY( 10 );
|
|
|
|
} while( !u12io_CheckTimer( &timer ));
|
|
|
|
u12io_DataToRegister( dev, REG_STEPCONTROL, _MOTOR0_SCANSTATE );
|
|
}
|
|
|
|
/** move the sensor to the appropriate shading position
|
|
*/
|
|
static SANE_Status u12motor_GotoShadingPosition( U12_Device *dev )
|
|
{
|
|
SANE_Byte rb[20];
|
|
SANE_Status res;
|
|
int c;
|
|
|
|
DBG( _DBG_INFO, "u12motor_GotoShadingPosition()\n" );
|
|
res = u12motor_ModuleToHome( dev );
|
|
if( SANE_STATUS_GOOD == res )
|
|
return res;
|
|
|
|
/* position to somewhere under the transparency adapter */
|
|
if( dev->DataInf.dwScanFlag & _SCANDEF_TPA ) {
|
|
|
|
u12motor_ForceToLeaveHomePos( dev );
|
|
u12motor_DownloadNullScanStates( dev );
|
|
|
|
c = 0;
|
|
_SET_REG( rb, c, REG_STEPCONTROL, _MOTOR0_SCANSTATE );
|
|
_SET_REG( rb, c, REG_MODECONTROL, _ModeScan);
|
|
_SET_REG( rb, c, REG_MOTOR0CONTROL, _FORWARD_MOTOR );
|
|
_SET_REG( rb, c, REG_XSTEPTIME, 6);
|
|
_SET_REG( rb, c, REG_EXTENDEDXSTEP, 0);
|
|
_SET_REG( rb, c, REG_SCANCONTROL1, _MFRC_BY_XSTEP);
|
|
u12io_DataToRegs( dev, rb, c );
|
|
|
|
res = u12motor_PositionYProc( dev, _TPA_SHADINGORG );
|
|
if( SANE_STATUS_GOOD != res )
|
|
return res;
|
|
}
|
|
DBG( _DBG_INFO, "* Position reached\n" );
|
|
return SANE_STATUS_GOOD;
|
|
}
|
|
|
|
/**
|
|
*/
|
|
static void u12motor_ModuleForwardBackward( U12_Device *dev )
|
|
{
|
|
DBG( _DBG_INFO, "u12motor_ModuleForwardBackward()\n" );
|
|
|
|
switch( dev->scan.bModuleState ) {
|
|
|
|
case _MotorInNormalState:
|
|
DBG( _DBG_INFO, "* _MotorInNormalState\n" );
|
|
dev->scan.bModuleState = _MotorGoBackward;
|
|
u12io_DataToRegister( dev, REG_SCANCONTROL1,
|
|
(SANE_Byte)(dev->regs.RD_ScanControl1 & ~_MFRC_RUNSCANSTATE));
|
|
u12io_DataToRegister( dev, REG_MOTOR0CONTROL,
|
|
(SANE_Byte)(dev->regs.RD_Motor0Control & ~_MotorDirForward));
|
|
|
|
u12motor_ModuleFreeRun( dev, _BACKSTEPS );
|
|
u12io_StartTimer( &u12motor_Timer, (15 * _MSECOND));
|
|
break;
|
|
|
|
case _MotorGoBackward:
|
|
DBG( _DBG_INFO, "* _MotorGoBackward\n" );
|
|
if( u12io_CheckTimer( &u12motor_Timer)) {
|
|
if(!(u12io_GetExtendedStatus( dev ) & _STILL_FREE_RUNNING )) {
|
|
dev->scan.bModuleState = _MotorInStopState;
|
|
u12io_StartTimer( &u12motor_Timer, (50 *_MSECOND));
|
|
}
|
|
}
|
|
break;
|
|
|
|
case _MotorInStopState:
|
|
DBG( _DBG_INFO, "* _MotorInStopState\n" );
|
|
if( u12io_CheckTimer( &u12motor_Timer )) {
|
|
|
|
if( u12io_GetFifoLength( dev ) < dev->scan.dwMaxReadFifo ) {
|
|
dev->scan.bModuleState = _MotorAdvancing;
|
|
u12io_DataToRegister( dev, REG_SCANCONTROL1,
|
|
dev->regs.RD_ScanControl1);
|
|
u12io_DataToRegister( dev, REG_MOTOR0CONTROL,
|
|
dev->regs.RD_Motor0Control);
|
|
u12motor_ModuleFreeRun( dev, _FORWARDSTEPS );
|
|
u12io_StartTimer( &u12motor_Timer, (15 * _MSECOND));
|
|
}
|
|
}
|
|
break;
|
|
|
|
case _MotorAdvancing:
|
|
DBG( _DBG_INFO, "* _MotorAdvancing\n" );
|
|
if( u12io_CheckTimer( &u12motor_Timer)) {
|
|
if( !(u12io_GetScanState( dev ) & _SCANSTATE_STOP))
|
|
dev->scan.bModuleState = _MotorInNormalState;
|
|
else {
|
|
if (!(u12io_GetExtendedStatus( dev ) & _STILL_FREE_RUNNING )) {
|
|
u12io_RegisterToScanner( dev, REG_REFRESHSCANSTATE );
|
|
dev->scan.bModuleState = _MotorInNormalState;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* END U12-MOTOR.C ..........................................................*/
|