kopia lustrzana https://gitlab.com/sane-project/backends
8777 wiersze
193 KiB
C
8777 wiersze
193 KiB
C
/**
|
|
Copyright (C) 2001 Stéphane Voltz <svoltz@wanadoo.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 implements a SANE backend for Umax PP flatbed scanners. */
|
|
|
|
|
|
|
|
|
|
|
|
#include "../include/sane/config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_SYS_TIME_H
|
|
#include <sys/time.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_FCNTL_H
|
|
#include <fcntl.h>
|
|
#endif
|
|
#include "../include/sane/sanei_debug.h"
|
|
#include <errno.h>
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
#include <dev/ppbus/ppi.h>
|
|
#include <dev/ppbus/ppbconf.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
#include <sys/ioctl.h>
|
|
#include <linux/parport.h>
|
|
#include <linux/ppdev.h>
|
|
#endif
|
|
|
|
#if HAVE_SYS_IO_H && defined __GNUC__
|
|
# include <sys/io.h> /* GNU libc based Linux */
|
|
#elif HAVE_ASM_IO_H && defined __i386__
|
|
# include <asm/io.h> /* older Linux */
|
|
#elif HAVE_SYS_HW_H
|
|
# include <sys/hw.h> /* OS/2 */
|
|
#else
|
|
#ifdef _ICC
|
|
|
|
static inline unsigned char
|
|
inb (unsigned int port)
|
|
{
|
|
unsigned char ret;
|
|
|
|
__asm__ __volatile__ ("inb %1,%0":"=a" (ret):"d" ((u_int) port));
|
|
return ret;
|
|
}
|
|
|
|
static inline void
|
|
outb (unsigned int port, unsigned char value)
|
|
{
|
|
__asm__ __volatile__ ("outb %0,%1"::"a" (value), "d" ((u_int) port));
|
|
}
|
|
|
|
static inline void
|
|
insb (unsigned int port, void *addr, unsigned long count)
|
|
{
|
|
__asm__ __volatile__ ("rep ; insb":"=D" (addr), "=c" (count):"d" (port),
|
|
"0" (addr), "1" (count));
|
|
}
|
|
|
|
static inline void
|
|
insl (unsigned int port, void *addr, unsigned long count)
|
|
{
|
|
__asm__ __volatile__ ("rep ; insl":"=D" (addr), "=c" (count):"d" (port),
|
|
"0" (addr), "1" (count));
|
|
}
|
|
|
|
static inline void
|
|
outsb (unsigned int port, const void *addr, unsigned long count)
|
|
{
|
|
__asm__ __volatile__ ("rep ; outsb":"=S" (addr), "=c" (count):"d" (port),
|
|
"0" (addr), "1" (count));
|
|
}
|
|
|
|
static inline void
|
|
outsw (unsigned int port, const void *addr, unsigned long count)
|
|
{
|
|
__asm__ __volatile__ ("rep ; outsw":"=S" (addr), "=c" (count):"d" (port),
|
|
"0" (addr), "1" (count));
|
|
}
|
|
|
|
#else
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#warning "ENABLE_PARPORT_DIRECTIO overriden"
|
|
#undef ENABLE_PARPORT_DIRECTIO
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
/* we need either direct io or ppdev */
|
|
#if ! defined ENABLE_PARPORT_DIRECTIO && ! defined HAVE_LINUX_PPDEV_H && ! defined HAVE_DEV_PPBUS_PPI_H
|
|
#define IO_SUPPORT_MISSING
|
|
#endif
|
|
|
|
|
|
#include "umax_pp_low.h"
|
|
|
|
#ifdef DMALLOC
|
|
#include "dmalloc.h"
|
|
#endif
|
|
|
|
#ifndef __IO__
|
|
#define __IO__
|
|
|
|
#define DATA gPort+0x00
|
|
#define STATUS gPort+0x01
|
|
#define CONTROL gPort+0x02
|
|
#define EPPADR gPort+0x03
|
|
#define EPPDATA gPort+0x04
|
|
|
|
#define ECPDATA gPort+0x400
|
|
#define ECPCONTROL gPort+0x402
|
|
|
|
|
|
#endif
|
|
|
|
static int Fonc001 (void);
|
|
static int FoncSendWord (int *cmd);
|
|
|
|
static void SetEPPMode (int mode);
|
|
static int GetEPPMode (void);
|
|
static void SetModel (int model);
|
|
static int GetModel (void);
|
|
static int RingScanner (int count, unsigned long delay);
|
|
static int TestVersion (int no);
|
|
|
|
static int SendCommand (int cmd);
|
|
static void SPPResetLPT (void);
|
|
static int SendWord (int *cmd);
|
|
static int SendData (int *cmd, int len);
|
|
static int ReceiveData (int *cmd, int len);
|
|
static int SendLength (int *cmd, int len);
|
|
|
|
|
|
static void Init001 (void);
|
|
static int Init002 (int arg);
|
|
static int Init005 (int arg);
|
|
static int Init021 (void);
|
|
static int Init022 (void);
|
|
|
|
static void NibbleReadBuffer (int size, unsigned char *dest);
|
|
static void WriteBuffer (int size, unsigned char *source);
|
|
static void EPPReadBuffer (int size, unsigned char *dest);
|
|
static void EPPWriteBuffer (int size, unsigned char *source);
|
|
static void EPPRead32Buffer (int size, unsigned char *dest);
|
|
static int PausedReadBuffer (int size, unsigned char *dest);
|
|
static void EPPWrite32Buffer (int size, unsigned char *source);
|
|
|
|
|
|
static int SlowNibbleRegisterRead (int reg);
|
|
static int EPPRegisterRead (int reg);
|
|
static int ECPRegisterRead (int reg);
|
|
static void ClearRegister (int reg);
|
|
static void WriteSlow (int reg, int value);
|
|
static void EPPRegisterWrite (int reg, int value);
|
|
static void ECPRegisterWrite (int reg, int value);
|
|
|
|
static int Prologue (void);
|
|
static int Epilogue (void);
|
|
|
|
static int CmdSet (int cmd, int len, int *buffer);
|
|
static int CmdGet (int cmd, int len, int *buffer);
|
|
static int CmdSetGet (int cmd, int len, int *buffer);
|
|
|
|
|
|
static int CmdGetBuffer (int cmd, int len, unsigned char *buffer);
|
|
static int CmdGetBuffer32 (int cmd, int len, unsigned char *buffer);
|
|
static int CmdGetBlockBuffer (int cmd, int len, int window,
|
|
unsigned char *buffer);
|
|
|
|
static void Bloc2Decode (int *op);
|
|
static void Bloc8Decode (int *op);
|
|
void sanei_umax_pp_gamma (int *red, int *green, int *blue);
|
|
|
|
|
|
#define WRITESLOW(x,y) \
|
|
WriteSlow((x),(y)); \
|
|
DBG(16,"WriteSlow(0x%X,0x%X) passed... (%s:%d)\n",(x),(y),__FILE__,__LINE__);
|
|
|
|
#define SLOWNIBBLEREGISTEREAD(x,y) \
|
|
tmp=SlowNibbleRegisterRead(x);\
|
|
if(tmp!=y)\
|
|
{\
|
|
DBG(0,"SlowNibbleRegisterRead: found 0x%X expected 0x%X (%s:%d)\n",tmp,y,__FILE__,__LINE__);\
|
|
/*return 0;*/ \
|
|
}\
|
|
DBG(16,"SlowNibbleRegisterRead(0x%X)=0x%X passed... (%s:%d)\n",x,y,__FILE__,__LINE__);
|
|
|
|
|
|
#define REGISTERWRITE(x,y) \
|
|
RegisterWrite((x),(y)); \
|
|
DBG(16,"RegisterWrite(0x%X,0x%X) passed... (%s:%d)\n",(x),(y),__FILE__,__LINE__);
|
|
|
|
#define REGISTERREAD(x,y) \
|
|
tmp=RegisterRead(x);\
|
|
if(tmp!=y)\
|
|
{\
|
|
DBG(0,"RegisterRead, found 0x%X expected 0x%X (%s:%d)\n",tmp,y,__FILE__,__LINE__);\
|
|
return 0;\
|
|
}\
|
|
DBG(16,"RegisterRead(0x%X)=0x%X passed... (%s:%d)\n",x,y,__FILE__,__LINE__);
|
|
|
|
|
|
#define TRACE(level,msg) DBG(level, msg" (%s:%d)\n",__FILE__,__LINE__);
|
|
|
|
|
|
#define CMDSYNC(x) if(sanei_umax_pp_CmdSync(x)!=1)\
|
|
{\
|
|
DBG(0,"CmdSync(0x%02X) failed (%s:%d)\n",x,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdSync() passed ...")
|
|
|
|
#define CMDSETGET(cmd,len,sent) if(CmdSetGet(cmd,len,sent)!=1)\
|
|
{\
|
|
DBG(0,"CmdSetGet(0x%02X,%d,sent) failed (%s:%d)\n",cmd,len,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdSetGet() passed ...")
|
|
|
|
#define YOFFSET 40
|
|
#define YOFFSET1220P 40
|
|
#define YOFFSET2000P 40
|
|
|
|
|
|
|
|
#define COMPLETIONWAIT if(CompletionWait()==0)\
|
|
{\
|
|
DBG(0,"CompletionWait() failed (%s:%d)\n",__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CompletionWait() passed ...")
|
|
|
|
#define MOVE(x,y,t) if(Move(x,y,t)==0)\
|
|
{\
|
|
DBG(0,"Move(%d,%d,buffer) failed (%s:%d)\n",x,y,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"Move() passed ...")
|
|
|
|
#define CMDGETBUF(cmd,len,sent) if(CmdGetBuffer(cmd,len,sent)!=1)\
|
|
{\
|
|
DBG(0,"CmdGetBuffer(0x%02X,%ld,buffer) failed (%s:%d)\n",cmd,(long)len,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdGetBuffer() passed ...")
|
|
|
|
#define CMDGETBUF32(cmd,len,sent) if(CmdGetBuffer32(cmd,len,sent)!=1)\
|
|
{\
|
|
DBG(0,"CmdGetBuffer32(0x%02X,%ld,buffer) failed (%s:%d)\n",cmd,(long)len,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdGetBuffer32() passed ...")
|
|
|
|
#define CMDSET(cmd,len,sent) if(CmdSet(cmd,len,sent)!=1)\
|
|
{\
|
|
DBG(0,"CmdSet(0x%02X,%d,sent) failed (%s:%d)\n",cmd,len,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdSet() passed ...")
|
|
|
|
#define CMDGET(cmd,len,sent) if(CmdGet(cmd,len,sent)!=1)\
|
|
{\
|
|
DBG(0,"CmdGet(0x%02X,%d,read) failed (%s:%d)\n",cmd,len,__FILE__,__LINE__);\
|
|
return(0);\
|
|
}\
|
|
TRACE(16,"CmdGet() passed ...")
|
|
|
|
|
|
|
|
static int gPort = 0x378;
|
|
|
|
/* global control vars */
|
|
static int gControl = 0;
|
|
static int gData = 0;
|
|
static int g674 = 0; /* semble indiquer qu'on utilise les IRQ */
|
|
static int g67D = 0;
|
|
static int g67E = 0;
|
|
static int gEPAT = 0; /* signals fast mode ? */
|
|
static int g6FE = 0;
|
|
|
|
/* default gamma translation table */
|
|
static int ggamma[256] =
|
|
{ 0x00, 0x06, 0x0A, 0x0D, 0x10, 0x12, 0x14, 0x17, 0x19, 0x1B, 0x1D, 0x1F,
|
|
0x21, 0x23, 0x24, 0x26, 0x28, 0x2A, 0x2B, 0x2D, 0x2E, 0x30, 0x31, 0x33,
|
|
0x34, 0x36, 0x37, 0x39, 0x3A, 0x3B, 0x3D, 0x3E, 0x40, 0x41, 0x42, 0x43,
|
|
0x45, 0x46, 0x47, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4F, 0x50, 0x51, 0x52,
|
|
0x53, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5E, 0x5F, 0x60,
|
|
0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C,
|
|
0x6D, 0x6E, 0x6F, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78,
|
|
0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F, 0x80, 0x81, 0x82, 0x83, 0x84,
|
|
0x85, 0x86, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
0x90, 0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x97, 0x98, 0x99,
|
|
0x9A, 0x9B, 0x9C, 0x9D, 0x9D, 0x9E, 0x9F, 0xA0, 0xA1, 0xA2, 0xA2, 0xA3,
|
|
0xA4, 0xA5, 0xA6, 0xA7, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAC, 0xAD,
|
|
0xAE, 0xAF, 0xB0, 0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB4, 0xB5, 0xB6, 0xB7,
|
|
0xB8, 0xB8, 0xB9, 0xBA, 0xBB, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF, 0xBF, 0xC0,
|
|
0xC1, 0xC2, 0xC2, 0xC3, 0xC4, 0xC5, 0xC5, 0xC6, 0xC7, 0xC8, 0xC8, 0xC9,
|
|
0xCA, 0xCB, 0xCB, 0xCC, 0xCD, 0xCE, 0xCE, 0xCF, 0xD0, 0xD1, 0xD1, 0xD2,
|
|
0xD3, 0xD4, 0xD4, 0xD5, 0xD6, 0xD6, 0xD7, 0xD8, 0xD9, 0xD9, 0xDA, 0xDB,
|
|
0xDC, 0xDC, 0xDD, 0xDE, 0xDE, 0xDF, 0xE0, 0xE1, 0xE1, 0xE2, 0xE3, 0xE3,
|
|
0xE4, 0xE5, 0xE6, 0xE6, 0xE7, 0xE8, 0xE8, 0xE9, 0xEA, 0xEA, 0xEB, 0xEC,
|
|
0xEC, 0xED, 0xEE, 0xEF, 0xEF, 0xF0, 0xF1, 0xF1, 0xF2, 0xF3, 0xF3, 0xF4,
|
|
0xF5, 0xF5, 0xF6, 0xF7, 0xF7, 0xF8, 0xF9, 0xF9, 0xFA, 0xFB, 0xFB, 0xFC,
|
|
0xFD, 0xFE, 0xFE, 0xFF
|
|
};
|
|
|
|
|
|
/* default gamma translation table */
|
|
static int *ggGreen = ggamma;
|
|
static int *ggBlue = ggamma;
|
|
static int *ggRed = ggamma;
|
|
static int gParport = 0;
|
|
static int gCancel = 0;
|
|
static int gAutoSettings = 1;
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* output one byte on given port */
|
|
/*****************************************************************************/
|
|
static void Outb (int port, int value);
|
|
|
|
/*****************************************************************************/
|
|
/* ouput 'size' bytes stored in 'source' on given port */
|
|
/*****************************************************************************/
|
|
static void Outsb (int port, unsigned char *source, int size);
|
|
|
|
/*****************************************************************************/
|
|
/* ouput 'size' 32 bits words stored in 'source' on given port */
|
|
/*****************************************************************************/
|
|
static void Outsw (int port, unsigned char *source, int size);
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* input one byte from given port */
|
|
/*****************************************************************************/
|
|
static int Inb (int port);
|
|
|
|
/*****************************************************************************/
|
|
/* input 'size' bytes from given port ans store them in 'dest' */
|
|
/*****************************************************************************/
|
|
static void Insb (int port, unsigned char *dest, int size);
|
|
|
|
/*****************************************************************************/
|
|
/* input 'size' 32 bits word from given port ans store them in 'dest' */
|
|
/*****************************************************************************/
|
|
static void Insw (int port, unsigned char *dest, int size);
|
|
|
|
|
|
/*
|
|
* returns 1 if succeds in getting base addr via /proc
|
|
* 0 on failure
|
|
*
|
|
* on successfull return, *addr will hold parport base addr
|
|
*/
|
|
int
|
|
sanei_parport_info (int number, int *addr)
|
|
{
|
|
char name[256];
|
|
FILE *fic = NULL;
|
|
char buffer[64], val[16];
|
|
int baseadr, ecpadr;
|
|
|
|
/* try 2.4 first */
|
|
sprintf (name, "/proc/sys/dev/parport/parport%d/base-addr", number);
|
|
memset (buffer, 0, 64);
|
|
memset (val, 0, 16);
|
|
fic = fopen (name, "rb");
|
|
if (fic == NULL)
|
|
{
|
|
/* open failure, try 2.2 */
|
|
sprintf (name, "/proc/parport/%d/hardware", number);
|
|
fic = fopen (name, "rb");
|
|
if (fic == NULL)
|
|
{ /* no proc at all */
|
|
DBG (1, "sanei_parport_info(): no /proc \n");
|
|
return (0);
|
|
}
|
|
fread (buffer, 64, 1, fic);
|
|
fclose (fic);
|
|
sscanf (buffer, "base: %s", val);
|
|
baseadr = strtol (val, NULL, 16);
|
|
|
|
}
|
|
else
|
|
{
|
|
fread (buffer, 64, 1, fic);
|
|
fclose (fic);
|
|
if (sscanf (buffer, "%d %d", &baseadr, &ecpadr) < 1)
|
|
{
|
|
/* empty base file */
|
|
return (0);
|
|
}
|
|
*addr = baseadr;
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
* gain direct acces to IO port, and set parport to the 'right' mode
|
|
* returns 1 on success, 0 an failure
|
|
*/
|
|
|
|
|
|
int
|
|
sanei_umax_pp_InitPort (int port, char *name)
|
|
{
|
|
int fd, ectr;
|
|
int found = 0, ecp = 1;
|
|
#if ((defined HAVE_IOPERM)||(defined HAVE_LINUX_PPDEV_H))
|
|
int mode;
|
|
#endif
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
char strmodes[160];
|
|
#endif
|
|
|
|
/* since this function must be called before */
|
|
/* any other, we put debug init here */
|
|
DBG_INIT ();
|
|
|
|
/* sets global vars */
|
|
ggGreen = ggamma;
|
|
ggBlue = ggamma;
|
|
ggRed = ggamma;
|
|
gParport = 0;
|
|
gCancel = 0;
|
|
gAutoSettings = 1;
|
|
gControl = 0;
|
|
gData = 0;
|
|
g674 = 0;
|
|
g67D = 0;
|
|
g67E = 0;
|
|
gEPAT = 0;
|
|
g6FE = 0;
|
|
sanei_umax_pp_setparport (0);
|
|
|
|
|
|
DBG (1, "sanei_umax_pp_InitPort(0x%X,%s)\n", port, name);
|
|
#ifndef ENABLE_PARPORT_DIRECTIO
|
|
if ((name == NULL) || ((name != NULL) && (strlen (name) < 4)))
|
|
{
|
|
DBG (0, "sanei_umax_pp_InitPort cannot use direct hardware access\n");
|
|
DBG (0, "if not compiled with --enable-parport-directio\n");
|
|
return (0);
|
|
}
|
|
#endif
|
|
|
|
|
|
/* init global var holding port value */
|
|
gPort = port;
|
|
|
|
#ifdef IO_SUPPORT_MISSING
|
|
DBG (1, "*** Direct I/O or ppdev unavailable, giving up ***\n");
|
|
return (0);
|
|
#else
|
|
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
if (name != NULL)
|
|
{
|
|
if (strlen (name) > 3)
|
|
{
|
|
/* ppdev opening and configuration */
|
|
found = 0;
|
|
fd = open (name, O_RDWR | O_NOCTTY | O_NONBLOCK);
|
|
if (fd < 0)
|
|
{
|
|
switch (errno)
|
|
{
|
|
case ENOENT:
|
|
DBG (1, "umax_pp: '%s' does not exist \n", name);
|
|
break;
|
|
case EACCES:
|
|
DBG (1,
|
|
"umax_pp: current user has not R/W permissions on '%s' \n",
|
|
name);
|
|
break;
|
|
}
|
|
return (0);
|
|
|
|
}
|
|
/* claim port */
|
|
if (ioctl (fd, PPCLAIM))
|
|
{
|
|
DBG (1, "umax_pp: cannot claim port '%s'\n", name);
|
|
}
|
|
else
|
|
{
|
|
/* we check if parport does EPP or ECP */
|
|
#ifdef PPGETMODES
|
|
if (ioctl (fd, PPGETMODES, &mode))
|
|
{
|
|
DBG (16,
|
|
"umax_pp: ppdev couldn't gave modes for port '%s'\n",
|
|
name);
|
|
}
|
|
else
|
|
{
|
|
sprintf (strmodes, "\n");
|
|
if (mode & PARPORT_MODE_PCSPP)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_PCSPP\n",
|
|
strmodes);
|
|
if (mode & PARPORT_MODE_TRISTATE)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_TRISTATE\n",
|
|
strmodes);
|
|
if (mode & PARPORT_MODE_EPP)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_EPP\n", strmodes);
|
|
if (mode & PARPORT_MODE_ECP)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_ECP\n", strmodes);
|
|
if (mode & PARPORT_MODE_COMPAT)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_COMPAT\n",
|
|
strmodes);
|
|
if (mode & PARPORT_MODE_DMA)
|
|
sprintf (strmodes, "%s\t\tPARPORT_MODE_DMA\n", strmodes);
|
|
DBG (32, "parport modes: %X\n", mode);
|
|
DBG (32, "parport modes: %s\n", strmodes);
|
|
if (!(mode & PARPORT_MODE_EPP)
|
|
&& !(mode & PARPORT_MODE_ECP))
|
|
{
|
|
DBG (1,
|
|
"port 0x%X does not have EPP or ECP, giving up ...\n",
|
|
port);
|
|
mode = IEEE1284_MODE_COMPAT;
|
|
ioctl (fd, PPSETMODE, &mode);
|
|
ioctl (fd, PPRELEASE);
|
|
close (fd);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
#else
|
|
DBG (16,
|
|
"umax_pp: ppdev used to build SANE doesn't have PPGETMODES.\n");
|
|
#endif
|
|
/* prefered mode is EPP */
|
|
mode = IEEE1284_MODE_EPP;
|
|
mode = ioctl (fd, PPNEGOT, &mode);
|
|
if (mode)
|
|
{
|
|
DBG (16,
|
|
"umax_pp: ppdev couldn't negociate mode IEEE1284_MODE_EPP for '%s'\n",
|
|
name);
|
|
}
|
|
if (ioctl (fd, PPSETMODE, &mode))
|
|
{
|
|
DBG (16,
|
|
"umax_pp: ppdev couldn't set mode to IEEE1284_MODE_EPP for '%s'\n",
|
|
name);
|
|
|
|
mode = IEEE1284_MODE_ECP;
|
|
if (ioctl (fd, PPSETMODE, &mode))
|
|
{
|
|
DBG (16,
|
|
"umax_pp: ppdev couldn't set mode to IEEE1284_MODE_ECP for '%s'\n",
|
|
name);
|
|
DBG (1,
|
|
"port 0x%X can't be set to EPP or ECP, giving up ...\n",
|
|
port);
|
|
|
|
mode = IEEE1284_MODE_COMPAT;
|
|
ioctl (fd, PPSETMODE, &mode);
|
|
ioctl (fd, PPRELEASE);
|
|
close (fd);
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
ecp = 1;
|
|
DBG (16,
|
|
"umax_pp: mode set to PARPORT_MODE_ECP for '%s'\n",
|
|
name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DBG (16,
|
|
"umax_pp: mode set to PARPORT_MODE_EPP for '%s'\n",
|
|
name);
|
|
ecp = 0;
|
|
}
|
|
|
|
|
|
/* allways start in compat mode (for probe) */
|
|
mode = IEEE1284_MODE_COMPAT;
|
|
ioctl (fd, PPSETMODE, &mode);
|
|
found = 1;
|
|
|
|
}
|
|
|
|
if (!found)
|
|
{
|
|
DBG (1, "device %s does not fit ...\n", name);
|
|
}
|
|
else
|
|
{
|
|
DBG (1, "Using %s ...\n", name);
|
|
sanei_umax_pp_setparport (fd);
|
|
return (1);
|
|
}
|
|
}
|
|
}
|
|
#endif /* HAVE_LINUX_PPDEV_H */
|
|
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
/* the ppi device let user access to parallel port on freebsd */
|
|
if (name != NULL)
|
|
{
|
|
if (strlen (name) > 3)
|
|
{
|
|
/* ppi opening and configuration */
|
|
found = 0;
|
|
fd = open (name, O_RDONLY);
|
|
if (fd < 0)
|
|
{
|
|
switch (errno)
|
|
{
|
|
case ENOENT:
|
|
DBG (1, "umax_pp: '%s' does not exist \n", name);
|
|
break;
|
|
case EACCES:
|
|
DBG (1,
|
|
"umax_pp: current user has not read permissions on '%s' \n",
|
|
name);
|
|
break;
|
|
}
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
DBG (1, "Using %s ...\n", name);
|
|
sanei_umax_pp_setparport (fd);
|
|
/* set up ECPEPP the hard way ... */
|
|
/* frob_econtrol (port, 0xe0, 4 << 5);
|
|
unsigned char ectr = inb (ECONTROL (pb));
|
|
outb ((ectr & ~m) ^ v, ECONTROL (pb)); */
|
|
ectr = Inb (ECPCONTROL);
|
|
if (ectr != 0xFF)
|
|
{
|
|
ectr = (ectr & ~(0xE0)) ^ (4 << 5);
|
|
Outb (ECPCONTROL, ectr);
|
|
DBG (1, "Setting ECPEPP ...\n");
|
|
}
|
|
return (1);
|
|
}
|
|
}
|
|
}
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
|
|
|
|
#ifdef HAVE_SYS_HW_H
|
|
/* gainig io perm under OS/2 */
|
|
/* IOPL must has been raised to 3 in config.sys */
|
|
/* for EMX portaccess always succeeds */
|
|
if (!found)
|
|
{
|
|
_portaccess (port, port + 7);
|
|
return (1);
|
|
}
|
|
#endif
|
|
|
|
|
|
/* FreeBSD and NetBSD with compatibility opion 9 */
|
|
/* opening /dev/io raise IOPL to level 3 */
|
|
fd = open ("/dev/io", O_RDWR);
|
|
if (errno == EACCES)
|
|
{
|
|
/* /dev/io exist but process hasn't the right permission */
|
|
DBG (1, "/dev/io could not gain access to 0x%X\n", port);
|
|
return (0);
|
|
}
|
|
if ((errno == ENXIO) || (errno == ENOENT))
|
|
{
|
|
/* /dev/io does not exist */
|
|
DBG (16, "no '/dev/io' device\n");
|
|
}
|
|
else if (errno != 0)
|
|
{
|
|
/* /dev/io we get an unexpected error */
|
|
DBG (1, "opening '/dev/io' got unexpected errno=%d\n", errno);
|
|
return (0);
|
|
}
|
|
else
|
|
return (1);
|
|
|
|
#ifdef HAVE_IOPERM
|
|
if (port < 0x400)
|
|
{
|
|
if (ioperm (port, 8, 1) != 0)
|
|
{
|
|
DBG (1, "ioperm could not gain access to 0x%X\n", port);
|
|
return (0);
|
|
}
|
|
}
|
|
/* ECP i/o range */
|
|
if (iopl (3) != 0)
|
|
{
|
|
DBG (1, "iopl could not raise IO permission to level 3\n");
|
|
return (0);
|
|
}
|
|
|
|
/* set up ECPEPP the hard way ... */
|
|
/* frob_econtrol (port, 0xe0, 4 << 5);
|
|
unsigned char ectr = inb (ECONTROL (pb));
|
|
outb ((ectr & ~m) ^ v, ECONTROL (pb)); */
|
|
if (ecp)
|
|
{
|
|
ectr = Inb (ECPCONTROL);
|
|
if (ectr != 0xFF)
|
|
{
|
|
DBG (1, "Setting mode to ECPEPP\n");
|
|
ectr = (ectr & ~(0xE0)) ^ (4 << 5);
|
|
Outb (ECPCONTROL, ectr);
|
|
}
|
|
}
|
|
|
|
/* in case of suid, return to real user rights */
|
|
mode = getuid ();
|
|
setreuid (mode, mode);
|
|
mode = getgid ();
|
|
setregid (mode, mode);
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* IO_SUPPORT_MISSING */
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
Outb (int port, int value)
|
|
{
|
|
#ifndef IO_SUPPORT_MISSING
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, rc;
|
|
unsigned char val;
|
|
|
|
|
|
fd = sanei_umax_pp_getparport ();
|
|
val = (unsigned char) value;
|
|
if (fd > 0)
|
|
{
|
|
/* there should be ECPCONTROL that doesn't go through ppdev */
|
|
/* it will leave when all the I/O will be done with ppdev */
|
|
switch (port - gPort)
|
|
{
|
|
case 0:
|
|
rc = ioctl (fd, PPWDATA, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 2:
|
|
if (val & 0x20)
|
|
{
|
|
rc = ioctl (fd, PPDATADIR, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n",
|
|
strerror (errno), __FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
rc = ioctl (fd, PPWCONTROL, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n",
|
|
strerror (errno), __FILE__, __LINE__);
|
|
}
|
|
return;
|
|
case 0x400:
|
|
case 0x402:
|
|
break;
|
|
default:
|
|
DBG (16, "Outb(0x%03X,0x%02X) escaped ppdev\n", port, value);
|
|
}
|
|
}
|
|
#endif /* HAVE_LINUX_PPDEV_H */
|
|
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int fd, rc;
|
|
unsigned char val;
|
|
|
|
|
|
fd = sanei_umax_pp_getparport ();
|
|
val = (unsigned char) value;
|
|
if (fd > 0)
|
|
{
|
|
switch (port - gPort)
|
|
{
|
|
case 0:
|
|
rc = ioctl (fd, PPISDATA, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 1:
|
|
rc = ioctl (fd, PPISSTATUS, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 2:
|
|
rc = ioctl (fd, PPISCTRL, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 3:
|
|
rc = ioctl (fd, PPISEPPA, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 4:
|
|
rc = ioctl (fd, PPISEPPD, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
case 0x402:
|
|
rc = ioctl (fd, PPISECR, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return;
|
|
default:
|
|
DBG (16, "Outb(0x%03X,0x%02X) escaped ppi\n", port, value);
|
|
}
|
|
}
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifdef HAVE_SYS_HW_H
|
|
_outp8 (port, value);
|
|
#else
|
|
outb (value, port);
|
|
#endif /* HAVE_SYS_HW_H */
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
|
|
#endif /* IO_SUPPORT_MISSING */
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
Inb (int port)
|
|
{
|
|
int res = 0xFF;
|
|
#ifndef IO_SUPPORT_MISSING
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, rc;
|
|
unsigned char val;
|
|
|
|
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
/* there should be ECPCONTROL that doesn't go through ppdev */
|
|
/* it will leave when all the I/O will be done with ppdev */
|
|
switch (port - gPort)
|
|
{
|
|
case 0:
|
|
rc = ioctl (fd, PPRDATA, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
|
|
case 1:
|
|
rc = ioctl (fd, PPRSTATUS, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
|
|
case 2:
|
|
rc = ioctl (fd, PPRCONTROL, &val);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
|
|
case 0x400:
|
|
case 0x402:
|
|
break;
|
|
|
|
default:
|
|
DBG (16, "Inb(0x%03X) escaped ppdev\n", port);
|
|
}
|
|
}
|
|
#endif /* HAVE_LINUX_PPDEV_H */
|
|
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int fd, rc;
|
|
unsigned char val;
|
|
|
|
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
switch (port - gPort)
|
|
{
|
|
case 0:
|
|
rc = ioctl (fd, PPIGDATA, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
case 1:
|
|
rc = ioctl (fd, PPIGSTATUS, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
case 2:
|
|
rc = ioctl (fd, PPIGCTRL, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
case 3:
|
|
rc = ioctl (fd, PPIGEPPA, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
case 4:
|
|
rc = ioctl (fd, PPIGEPPD, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
case 0x402:
|
|
rc = ioctl (fd, PPIGECR, &val);
|
|
if (rc)
|
|
DBG (0, "ppi ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
res = val;
|
|
return res;
|
|
default:
|
|
DBG (16, "Inb(0x%03X) escaped ppi\n", port);
|
|
}
|
|
}
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifdef HAVE_SYS_HW_H
|
|
res = _inp8 (port) & 0xFF;
|
|
#else
|
|
res = inb (port) & 0xFF;
|
|
#endif /* HAVE_SYS_HW_H */
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
|
|
#endif /* IO_SUPPORT_MISSING */
|
|
return res;
|
|
}
|
|
|
|
|
|
static void
|
|
Insb (int port, unsigned char *dest, int size)
|
|
{
|
|
#ifndef IO_SUPPORT_MISSING
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int i;
|
|
for (i = 0; i < size; i++)
|
|
dest[i] = Inb (port);
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifndef __i386__
|
|
int i;
|
|
for (i = 0; i < size; i++)
|
|
dest[i] = Inb (port);
|
|
#else
|
|
#ifdef HAVE_SYS_HW_H
|
|
_inps8 (port, dest, size);
|
|
#else
|
|
insb (port, dest, size);
|
|
#endif
|
|
#endif
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
Outsb (int port, unsigned char *source, int size)
|
|
{
|
|
#ifndef IO_SUPPORT_MISSING
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int i;
|
|
|
|
for (i = 0; i < size; i++)
|
|
Outb (port, source[i]);
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifndef __i386__
|
|
int i;
|
|
|
|
for (i = 0; i < size; i++)
|
|
Outb (port, source[i]);
|
|
#else
|
|
#ifdef HAVE_SYS_HW_H
|
|
_outps8 (port, source, size);
|
|
#else
|
|
outsb (port, source, size);
|
|
#endif
|
|
#endif
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
/* size = nb words */
|
|
static void
|
|
Insw (int port, unsigned char *dest, int size)
|
|
{
|
|
#ifndef IO_SUPPORT_MISSING
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int i;
|
|
|
|
for (i = 0; i < size * 4; i++)
|
|
dest[i] = Inb (port);
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifndef __i386__
|
|
int i;
|
|
|
|
for (i = 0; i < size * 4; i++)
|
|
dest[i] = Inb (port);
|
|
#else
|
|
#ifdef HAVE_SYS_HW_H
|
|
_inps32 (port, (unsigned long *) dest, size);
|
|
#else
|
|
insl (port, dest, size);
|
|
#endif
|
|
#endif
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
Outsw (int port, unsigned char *source, int size)
|
|
{
|
|
#ifndef IO_SUPPORT_MISSING
|
|
|
|
#ifdef HAVE_DEV_PPBUS_PPI_H
|
|
int i;
|
|
|
|
for (i = 0; i < size * 4; i++)
|
|
Outb (port, source[i]);
|
|
#endif /* HAVE_DEV_PPBUS_PPI_H */
|
|
|
|
#ifdef ENABLE_PARPORT_DIRECTIO
|
|
#ifndef __i386__
|
|
int i;
|
|
|
|
for (i = 0; i < size * 4; i++)
|
|
Outb (port, source[i]);
|
|
#else
|
|
#ifdef HAVE_SYS_HW_H
|
|
_outps32 (port, (unsigned long *) source, size);
|
|
#else
|
|
outsw (port, source, size);
|
|
#endif
|
|
#endif
|
|
#endif /* ENABLE_PARPORT_DIRECTIO */
|
|
#endif
|
|
}
|
|
|
|
|
|
/* we're trying to gather information on the scanner here, */
|
|
/* and published it through an easy interface */
|
|
static int scannerStatus = 0;
|
|
static int epp32 = 1;
|
|
static int gMode = UMAX_PP_PARPORT_SPP;
|
|
static int model = 0x15;
|
|
static int astra = 0;
|
|
static int hasUTA = 0;
|
|
|
|
int
|
|
sanei_umax_pp_UTA (void)
|
|
{
|
|
return (hasUTA);
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_ScannerStatus (void)
|
|
{
|
|
/* 0x07 variant returns status with bit 0 or 1 allways set to 1 */
|
|
/* so we mask it out */
|
|
return (scannerStatus & 0xFC);
|
|
}
|
|
|
|
static int
|
|
GetEPPMode (void)
|
|
{
|
|
if (epp32)
|
|
return (32);
|
|
return (8);
|
|
}
|
|
|
|
static void
|
|
SetEPPMode (int mode)
|
|
{
|
|
if (mode == 8)
|
|
epp32 = 0;
|
|
else
|
|
epp32 = 1;
|
|
}
|
|
|
|
static int
|
|
GetModel (void)
|
|
{
|
|
return (model);
|
|
}
|
|
|
|
static void
|
|
SetModel (int mod)
|
|
{
|
|
model = mod;
|
|
}
|
|
|
|
|
|
int
|
|
sanei_umax_pp_getparport (void)
|
|
{
|
|
return (gParport);
|
|
}
|
|
|
|
void
|
|
sanei_umax_pp_setparport (int fd)
|
|
{
|
|
gParport = fd;
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_getport (void)
|
|
{
|
|
return (gPort);
|
|
}
|
|
|
|
void
|
|
sanei_umax_pp_setport (int port)
|
|
{
|
|
gPort = port;
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_getastra (void)
|
|
{
|
|
return (astra);
|
|
}
|
|
|
|
void
|
|
sanei_umax_pp_setastra (int mod)
|
|
{
|
|
astra = mod;
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_getauto (void)
|
|
{
|
|
return gAutoSettings;
|
|
}
|
|
|
|
void
|
|
sanei_umax_pp_setauto (int autoset)
|
|
{
|
|
gAutoSettings = autoset;
|
|
}
|
|
|
|
|
|
static int
|
|
NibbleRead (void)
|
|
{
|
|
int res;
|
|
int tmp;
|
|
|
|
res = Inb (STATUS);
|
|
res = Inb (STATUS);
|
|
res = res & 0xF0;
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
|
|
tmp = Inb (STATUS);
|
|
tmp = Inb (STATUS);
|
|
tmp = (tmp & 0xF0) >> 4;
|
|
res = res | tmp;
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 5);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
/******************************************************************************/
|
|
/* WriteSlow: write value in register, slow method */
|
|
/******************************************************************************/
|
|
static void
|
|
WriteSlow (int reg, int value)
|
|
{
|
|
/* select register */
|
|
Outb (DATA, reg | 0x60);
|
|
Outb (DATA, reg | 0x60);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x01);
|
|
|
|
/* send value */
|
|
Outb (DATA, value);
|
|
Outb (DATA, value);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* Send command returns 0 on failure, 1 on success */
|
|
/*****************************************************************************/
|
|
static int
|
|
SendCommand (int cmd)
|
|
{
|
|
int control;
|
|
int tmp;
|
|
int val;
|
|
int i;
|
|
int gReadBuffer[256]; /* read buffer for command 0x10 */
|
|
|
|
|
|
if (g674 != 0)
|
|
{
|
|
DBG (0, "No scanner attached, SendCommand(0x%X) failed\n", cmd);
|
|
return 0;
|
|
}
|
|
|
|
control = Inb (CONTROL) & 0x3F;
|
|
tmp = cmd & 0xF8;
|
|
|
|
|
|
if ((g67D != 1) && (tmp != 0xE0) && (tmp != 0x20) && (tmp != 0x40)
|
|
&& (tmp != 0xD0) && (tmp != 0x08) && (tmp != 0x48))
|
|
{
|
|
Outb (CONTROL, 4); /* reset printer */
|
|
}
|
|
else
|
|
{
|
|
val = control & 0x1F;
|
|
if (g67D != 1)
|
|
val = val & 0xF; /* no IRQ */
|
|
val = val | 0x4;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
}
|
|
|
|
/* send sequence */
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, cmd);
|
|
Outb (DATA, cmd);
|
|
|
|
cmd = cmd & 0xF8;
|
|
|
|
if ((g67D == 1) && (cmd == 0xE0))
|
|
{
|
|
val = Inb (CONTROL);
|
|
val = (val & 0xC) | 0x01;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
val = val & 0xC;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
goto SendCommandEnd;
|
|
}
|
|
|
|
if ((cmd != 8) && (cmd != 0x48))
|
|
{
|
|
tmp = Inb (CONTROL);
|
|
tmp = Inb (CONTROL);
|
|
tmp = tmp & 0x1E;
|
|
if (g67D != 1)
|
|
tmp = tmp & 0xE;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
}
|
|
|
|
if (cmd == 0x10)
|
|
{
|
|
tmp = NibbleRead ();
|
|
tmp = tmp * 256 + NibbleRead ();
|
|
goto SendCommandEnd;
|
|
}
|
|
|
|
if (cmd == 8)
|
|
{
|
|
if (g67D == 1)
|
|
{
|
|
i = 0;
|
|
while (i < g67E)
|
|
{
|
|
tmp = Inb (CONTROL);
|
|
tmp = Inb (CONTROL);
|
|
tmp = (tmp & 0x1E) | 0x1;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
gReadBuffer[i] = Inb (STATUS);
|
|
tmp = tmp & 0x1E;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
|
|
/* next read */
|
|
i++;
|
|
if (i < g67E)
|
|
{
|
|
Outb (DATA, i | 8);
|
|
Outb (DATA, i | 8);
|
|
}
|
|
}
|
|
goto SendCommandEnd;
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "UNEXPLORED BRANCH %s:%d\n", __FILE__, __LINE__);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* */
|
|
if (cmd == 0)
|
|
{
|
|
i = 0;
|
|
do
|
|
{
|
|
tmp = Inb (CONTROL);
|
|
tmp = (tmp & 0xE) | 0x1;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
tmp = tmp & 0xE;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
|
|
i++;
|
|
if (i < g67E)
|
|
{
|
|
Outb (DATA, i);
|
|
Outb (DATA, i);
|
|
}
|
|
}
|
|
while (i < g67E);
|
|
goto SendCommandEnd;
|
|
}
|
|
|
|
if (cmd == 0x48)
|
|
{
|
|
/* this case is unneeded, should fit with the rest */
|
|
tmp = Inb (CONTROL) & 0x1E;
|
|
if (g67D != 1)
|
|
tmp = tmp & 0xE;
|
|
Outb (CONTROL, tmp | 0x1);
|
|
Outb (CONTROL, tmp | 0x1);
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
goto SendCommandEnd;
|
|
}
|
|
|
|
/* */
|
|
tmp = Inb (CONTROL) & 0x1E;
|
|
if (g67D != 1)
|
|
tmp = tmp & 0xE;
|
|
Outb (CONTROL, tmp | 0x1);
|
|
Outb (CONTROL, tmp | 0x1);
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
|
|
/* success */
|
|
SendCommandEnd:
|
|
|
|
if (cmd == 0x48)
|
|
Outb (CONTROL, (control & 0xF) | 0x4);
|
|
if (cmd == 0x30)
|
|
Outb (CONTROL, (gControl & 0xF) | 0x4);
|
|
|
|
/* end signature */
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
|
|
if (cmd == 8)
|
|
{
|
|
Outb (CONTROL, control);
|
|
return 1;
|
|
}
|
|
|
|
if (cmd == 0x30)
|
|
{
|
|
Outb (CONTROL, gControl);
|
|
return 1;
|
|
}
|
|
|
|
if (cmd != 0xE0)
|
|
Outb (CONTROL, control);
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
static void
|
|
ClearRegister (int reg)
|
|
{
|
|
|
|
/* choose register */
|
|
Outb (DATA, reg);
|
|
Outb (DATA, reg);
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 1);
|
|
if ((g6FE == 0) || (g674 != 0))
|
|
{
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 1);
|
|
}
|
|
|
|
/* clears it by not sending new value */
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
}
|
|
|
|
static void
|
|
SPPResetLPT (void)
|
|
{
|
|
Outb (CONTROL, 0x04);
|
|
}
|
|
|
|
|
|
static int
|
|
SlowNibbleRegisterRead (int reg)
|
|
{
|
|
int low, high;
|
|
|
|
|
|
/* send register number */
|
|
Outb (DATA, reg);
|
|
Outb (DATA, reg);
|
|
|
|
/* get low nibble */
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 3);
|
|
Outb (CONTROL, 3);
|
|
Outb (CONTROL, 3);
|
|
Outb (CONTROL, 3);
|
|
low = Inb (STATUS);
|
|
low = Inb (STATUS);
|
|
|
|
/* get high nibble */
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
high = Inb (STATUS);
|
|
high = Inb (STATUS);
|
|
|
|
/* merge nibbles and return */
|
|
high = (high & 0xF0) | ((low & 0xF0) >> 4);
|
|
return (high);
|
|
}
|
|
|
|
|
|
static void
|
|
NibbleReadBuffer (int size, unsigned char *dest)
|
|
{
|
|
int high;
|
|
int low;
|
|
int i;
|
|
int count;
|
|
int bytel, byteh;
|
|
|
|
/* init transfer */
|
|
Outb (DATA, 7);
|
|
Outb (DATA, 7);
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 1);
|
|
Outb (CONTROL, 3);
|
|
Outb (CONTROL, 3);
|
|
Outb (CONTROL, 3);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
count = (size - 2) / 2;
|
|
i = 0;
|
|
bytel = 0x06; /* signals low byte of word */
|
|
byteh = 0x07; /* signals high byte of word */
|
|
|
|
/* read loop */
|
|
while (count > 0)
|
|
{
|
|
/* low nibble byte 0 */
|
|
Outb (CONTROL, bytel);
|
|
Outb (CONTROL, bytel);
|
|
Outb (CONTROL, bytel);
|
|
low = Inb (STATUS);
|
|
if ((low & 0x08) == 0)
|
|
{
|
|
/* high nibble <> low nibble */
|
|
Outb (CONTROL, bytel & 0x05);
|
|
Outb (CONTROL, bytel & 0x05);
|
|
Outb (CONTROL, bytel & 0x05);
|
|
high = Inb (STATUS);
|
|
}
|
|
else
|
|
{
|
|
high = low;
|
|
}
|
|
low = low & 0xF0;
|
|
high = high & 0xF0;
|
|
dest[i] = (unsigned char) (high | (low >> 4));
|
|
i++;
|
|
|
|
/* low nibble byte 1 */
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
low = Inb (STATUS);
|
|
if ((low & 0x08) == 0)
|
|
{
|
|
/* high nibble <> low nibble */
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
high = Inb (STATUS);
|
|
}
|
|
else
|
|
{
|
|
high = low;
|
|
}
|
|
low = low & 0xF0;
|
|
high = high & 0xF0;
|
|
dest[i] = (unsigned char) (high | (low >> 4));
|
|
i++;
|
|
|
|
/* next read */
|
|
count--;
|
|
}
|
|
|
|
/* final read */
|
|
/* low nibble byte 0 */
|
|
Outb (CONTROL, bytel);
|
|
Outb (CONTROL, bytel);
|
|
Outb (CONTROL, bytel);
|
|
low = Inb (STATUS);
|
|
if ((low & 0x08) == 0)
|
|
{
|
|
/* high nibble <> low nibble */
|
|
Outb (CONTROL, bytel & 0x05);
|
|
Outb (CONTROL, bytel & 0x05);
|
|
Outb (CONTROL, bytel & 0x05);
|
|
high = Inb (STATUS);
|
|
}
|
|
else
|
|
{
|
|
high = low;
|
|
}
|
|
low = low & 0xF0;
|
|
high = high & 0xF0;
|
|
dest[i] = (unsigned char) (high | (low >> 4));
|
|
i++;
|
|
|
|
/* uneven size need an extra read */
|
|
if ((size & 0x01) == 1)
|
|
{
|
|
/* low nibble byte 1 */
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
low = Inb (STATUS);
|
|
if ((low & 0x08) == 0)
|
|
{
|
|
/* high nibble <> low nibble */
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
high = Inb (STATUS);
|
|
}
|
|
else
|
|
{
|
|
high = low;
|
|
}
|
|
low = low & 0xF0;
|
|
high = high & 0xF0;
|
|
dest[i] = (unsigned char) (high | (low >> 4));
|
|
i++;
|
|
|
|
/* swap high/low word */
|
|
count = bytel;
|
|
bytel = byteh;
|
|
byteh = count;
|
|
}
|
|
|
|
/* final byte ... */
|
|
Outb (DATA, 0xFD);
|
|
Outb (DATA, 0xFD);
|
|
Outb (DATA, 0xFD);
|
|
|
|
/* low nibble */
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
Outb (CONTROL, byteh);
|
|
low = Inb (STATUS);
|
|
if ((low & 0x08) == 0)
|
|
{
|
|
/* high nibble <> low nibble */
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
Outb (CONTROL, byteh & 0x05);
|
|
high = Inb (STATUS);
|
|
}
|
|
else
|
|
{
|
|
high = low;
|
|
}
|
|
low = low & 0xF0;
|
|
high = high & 0xF0;
|
|
dest[i] = (unsigned char) (high | (low >> 4));
|
|
i++;
|
|
|
|
/* reset port */
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (CONTROL, 0x04);
|
|
}
|
|
|
|
static void
|
|
WriteBuffer (int size, unsigned char *source)
|
|
{
|
|
int i;
|
|
int count;
|
|
int val;
|
|
|
|
/* init buffer write */
|
|
i = 0;
|
|
count = size / 2;
|
|
Outb (DATA, 0x67);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x05);
|
|
|
|
/* write loop */
|
|
while (count > 0)
|
|
{
|
|
/* low byte of word */
|
|
val = source[i];
|
|
i++;
|
|
Outb (DATA, val);
|
|
Outb (DATA, val);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
|
|
/* high byte of word */
|
|
val = source[i];
|
|
i++;
|
|
Outb (DATA, val);
|
|
Outb (DATA, val);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
|
|
/* next write */
|
|
count--;
|
|
}
|
|
|
|
/* termination sequence */
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x05);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
Init001 (void)
|
|
{
|
|
int i;
|
|
int val;
|
|
int status;
|
|
|
|
ClearRegister (0);
|
|
Outb (CONTROL, 0x0C);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (CONTROL, 0x0C);
|
|
}
|
|
Outb (DATA, 0x40);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (DATA, 0x40);
|
|
Outb (DATA, 0x40);
|
|
Outb (DATA, 0x40);
|
|
}
|
|
Outb (CONTROL, 0x06);
|
|
Outb (CONTROL, 0x06);
|
|
Outb (CONTROL, 0x06);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, 0x06);
|
|
Outb (CONTROL, 0x06);
|
|
Outb (CONTROL, 0x06);
|
|
}
|
|
|
|
/* sync loop */
|
|
i = 256;
|
|
do
|
|
{
|
|
status = Inb (STATUS);
|
|
i--;
|
|
}
|
|
while ((i > 0) && ((status & 0x40)));
|
|
val = 0x0C;
|
|
if (i > 0)
|
|
{
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
Outb (CONTROL, 0x07);
|
|
}
|
|
val = 0x04;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
}
|
|
}
|
|
val = val | 0x0C;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
}
|
|
val = val & 0xF7;
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
if (g674 != 0)
|
|
{
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
Outb (CONTROL, val);
|
|
}
|
|
}
|
|
|
|
/* SPP register reading */
|
|
static int
|
|
Init002 (int arg)
|
|
{
|
|
int data;
|
|
|
|
if (arg == 1)
|
|
return (0);
|
|
Outb (DATA, 0x0B);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (CONTROL, 0x24);
|
|
Outb (CONTROL, 0x24);
|
|
Outb (CONTROL, 0x26);
|
|
Outb (CONTROL, 0x26);
|
|
|
|
data = Inb (DATA);
|
|
Outb (CONTROL, 0x04);
|
|
if (data == gEPAT)
|
|
{
|
|
return (1);
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
ECPRegisterRead (int reg)
|
|
{
|
|
if (reg)
|
|
return 0xFF;
|
|
return 0xFF;
|
|
}
|
|
|
|
static int
|
|
EPPRegisterRead (int reg)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc;
|
|
unsigned char breg, bval;
|
|
#endif
|
|
int control;
|
|
int value;
|
|
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
breg = (unsigned char) (reg);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &breg, 1);
|
|
if (rc != 1)
|
|
DBG (0, "ppdev short write (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = read (fd, &bval, 1);
|
|
if (rc != 1)
|
|
DBG (0, "ppdev short read (%s:%d)\n", __FILE__, __LINE__);
|
|
value = bval;
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
|
|
return value;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
|
|
Outb (EPPADR, reg);
|
|
control = Inb (CONTROL);
|
|
control = (control & 0x1F) | 0x20;
|
|
Outb (CONTROL, control);
|
|
value = Inb (EPPDATA);
|
|
control = Inb (CONTROL);
|
|
control = control & 0x1F;
|
|
Outb (CONTROL, control);
|
|
return (value);
|
|
}
|
|
|
|
static int
|
|
RegisterRead (int reg)
|
|
{
|
|
switch (gMode)
|
|
{
|
|
case UMAX_PP_PARPORT_ECP:
|
|
return ECPRegisterRead (reg);
|
|
case UMAX_PP_PARPORT_EPP:
|
|
default:
|
|
return EPPRegisterRead (reg);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
ECPRegisterWrite (int reg, int value)
|
|
{
|
|
if ((reg) || (value))
|
|
return;
|
|
}
|
|
|
|
static void
|
|
EPPRegisterWrite (int reg, int value)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc;
|
|
unsigned char breg, bval;
|
|
#endif
|
|
|
|
reg = reg | 0x40;
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
breg = (unsigned char) (reg);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &breg, 1);
|
|
|
|
bval = (unsigned char) (value);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
return;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
Outb (EPPADR, reg);
|
|
Outb (EPPDATA, value);
|
|
}
|
|
|
|
static void
|
|
RegisterWrite (int reg, int value)
|
|
{
|
|
switch (gMode)
|
|
{
|
|
case UMAX_PP_PARPORT_ECP:
|
|
ECPRegisterWrite (reg, value);
|
|
break;
|
|
case UMAX_PP_PARPORT_EPP:
|
|
default:
|
|
EPPRegisterWrite (reg, value);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
EPPBlockMode (int flag)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc;
|
|
unsigned char bval;
|
|
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
bval = (unsigned char) (flag);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
return;
|
|
}
|
|
#endif
|
|
Outb (EPPADR, flag);
|
|
}
|
|
|
|
static void
|
|
EPPReadBuffer (int size, unsigned char *dest)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc, nb;
|
|
unsigned char bval;
|
|
#endif
|
|
int control;
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
|
|
bval = 0x80;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#ifdef PPSETFLAGS
|
|
mode = PP_FASTREAD;
|
|
rc = ioctl (fd, PPSETFLAGS, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#endif
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
nb = 0;
|
|
while (nb < size - 1)
|
|
{
|
|
rc = read (fd, dest + nb, size - 1 - nb);
|
|
nb += rc;
|
|
}
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
bval = 0xA0;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = read (fd, dest + size - 1, 1);
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
|
|
return;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
|
|
EPPBlockMode (0x80);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F) | 0x20); /* reverse */
|
|
Insb (EPPDATA, dest, size - 1);
|
|
control = Inb (CONTROL);
|
|
|
|
Outb (CONTROL, (control & 0x1F)); /* forward */
|
|
EPPBlockMode (0xA0);
|
|
control = Inb (CONTROL);
|
|
|
|
Outb (CONTROL, (control & 0x1F) | 0x20); /* reverse */
|
|
Insb (EPPDATA, (unsigned char *) (dest + size - 1), 1);
|
|
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F)); /* forward */
|
|
}
|
|
|
|
|
|
static void
|
|
EPPWriteBuffer (int size, unsigned char *source)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc;
|
|
unsigned char bval;
|
|
#endif
|
|
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
bval = 0xC0;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, source, size);
|
|
return;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
EPPBlockMode (0xC0);
|
|
Outsb (EPPDATA, source, size);
|
|
}
|
|
|
|
|
|
/* returns 0 if mode OK, else -1 */
|
|
static int
|
|
CheckEPAT (void)
|
|
{
|
|
int version;
|
|
|
|
version = RegisterRead (0x0B);
|
|
if (version == 0xC7)
|
|
return (0);
|
|
DBG (0, "CheckEPAT: expected EPAT version 0xC7, got 0x%X! (%s:%d)\n",
|
|
version, __FILE__, __LINE__);
|
|
return (-1);
|
|
|
|
}
|
|
|
|
static int
|
|
Init005 (int arg)
|
|
{
|
|
int count = 5;
|
|
int res;
|
|
|
|
while (count > 0)
|
|
{
|
|
RegisterWrite (0x0A, arg);
|
|
Outb (DATA, 0xFF);
|
|
res = RegisterRead (0x0A);
|
|
|
|
/* failed ? */
|
|
if (res != arg)
|
|
return (1);
|
|
|
|
/* ror arg */
|
|
res = arg & 0x01;
|
|
arg = arg / 2;
|
|
if (res == 1)
|
|
arg = arg | 0x80;
|
|
|
|
/* next loop */
|
|
count--;
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
|
|
static void
|
|
Init020 (void)
|
|
{
|
|
int control;
|
|
int data;
|
|
|
|
Outb (DATA, 0x04);
|
|
Outb (CONTROL, 0x0C);
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, control & 0x1F);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, control & 0x1F);
|
|
}
|
|
|
|
/* 1 OK, 0 failure */
|
|
static int
|
|
Init021 (void)
|
|
{
|
|
Init020 ();
|
|
if (SendCommand (0xE0) != 1)
|
|
{
|
|
DBG (0, "Init021: SendCommand(0xE0) failed! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
ClearRegister (0);
|
|
Init001 ();
|
|
return (1);
|
|
}
|
|
|
|
/* 1 OK, 0 failure */
|
|
static int
|
|
Init022 (void)
|
|
{
|
|
if (Init021 () != 1)
|
|
{
|
|
DBG (0, "Init022: Init021() failed! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
EPPRead32Buffer (int size, unsigned char *dest)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc, nb;
|
|
unsigned char bval;
|
|
#endif
|
|
int control;
|
|
|
|
if (GetEPPMode () == 8)
|
|
{
|
|
EPPReadBuffer (size, dest);
|
|
return;
|
|
}
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
|
|
bval = 0x80;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#ifdef PPSETFLAGS
|
|
mode = PP_FASTREAD;
|
|
rc = ioctl (fd, PPSETFLAGS, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#endif
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
nb = 0;
|
|
while (nb < size - 4)
|
|
{
|
|
rc = read (fd, dest + nb, size - 4 - nb);
|
|
nb += rc;
|
|
}
|
|
|
|
rc = read (fd, dest + size - 4, 3);
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
bval = 0xA0;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = read (fd, dest + size - 1, 1);
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
|
|
return;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
|
|
EPPBlockMode (0x80);
|
|
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F) | 0x20);
|
|
Insw (EPPDATA, dest, size / 4 - 1);
|
|
|
|
Insb (EPPDATA, (unsigned char *) (dest + size - 4), 3);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F));
|
|
|
|
EPPBlockMode (0xA0);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F) | 0x20);
|
|
|
|
Insb (EPPDATA, (unsigned char *) (dest + size - 1), 1);
|
|
control = Inb (CONTROL);
|
|
Outb (CONTROL, (control & 0x1F));
|
|
}
|
|
|
|
static void
|
|
EPPWrite32Buffer (int size, unsigned char *source)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
int fd, mode, rc;
|
|
unsigned char bval;
|
|
#endif
|
|
|
|
if (GetEPPMode () == 8)
|
|
{
|
|
EPPWriteBuffer (size, source);
|
|
return;
|
|
}
|
|
|
|
if ((size % 4) != 0)
|
|
{
|
|
DBG (0, "EPPWrite32Buffer: size %% 4 != 0!! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* check we have ppdev working */
|
|
fd = sanei_umax_pp_getparport ();
|
|
if (fd > 0)
|
|
{
|
|
|
|
bval = 0xC0;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
#ifdef PPSETFLAGS
|
|
mode = PP_FASTWRITE;
|
|
rc = ioctl (fd, PPSETFLAGS, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#endif
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, source, size);
|
|
|
|
return;
|
|
}
|
|
/* if not, direct hardware access */
|
|
#endif
|
|
EPPBlockMode (0xC0);
|
|
Outsw (EPPDATA, source, size / 4);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* returns 0 if ERROR cleared in STATUS within 1024 inb, else 1 */
|
|
static int
|
|
WaitOnError (void)
|
|
{
|
|
int c = 0;
|
|
int count = 1024;
|
|
int status;
|
|
|
|
do
|
|
{
|
|
do
|
|
{
|
|
status = Inb (STATUS) & 0x08;
|
|
if (status != 0)
|
|
{
|
|
count--;
|
|
if (count == 0)
|
|
c = 1;
|
|
}
|
|
}
|
|
while ((count > 0) && (status != 0));
|
|
if (status == 0)
|
|
{
|
|
status = Inb (STATUS) & 0x08;
|
|
if (status == 0)
|
|
c = 0;
|
|
}
|
|
}
|
|
while ((status != 0) && (c == 0));
|
|
return (c);
|
|
}
|
|
|
|
|
|
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
/* read up to size bytes, returns bytes read */
|
|
static int
|
|
ParportPausedReadBuffer (int size, unsigned char *dest)
|
|
{
|
|
unsigned char status, bval;
|
|
int error;
|
|
int word;
|
|
int bread;
|
|
int c;
|
|
int fd, rc, mode;
|
|
|
|
/* init */
|
|
bread = 0;
|
|
error = 0;
|
|
fd = sanei_umax_pp_getparport ();
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#ifdef PPSETFLAGS
|
|
mode = PP_FASTREAD;
|
|
rc = ioctl (fd, PPSETFLAGS, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
#endif
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
|
|
if ((size & 0x03) != 0)
|
|
{
|
|
while ((!error) && ((size & 0x03) != 0))
|
|
{
|
|
rc = read (fd, dest, 1);
|
|
size--;
|
|
dest++;
|
|
bread++;
|
|
rc = ioctl (fd, PPRSTATUS, &status);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
error = status & 0x08;
|
|
}
|
|
if (error)
|
|
{
|
|
DBG (0, "Read error (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
/* from here, we read 1 byte, then size/4-1 32 bits words, and then
|
|
3 bytes, pausing on ERROR bit of STATUS */
|
|
size -= 4;
|
|
|
|
/* sanity test, seems to be wrongly handled ... */
|
|
if (size == 0)
|
|
{
|
|
DBG (0, "case not handled! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
word = 0;
|
|
error = 0;
|
|
bread += size;
|
|
do
|
|
{
|
|
do
|
|
{
|
|
rc = read (fd, dest, 1);
|
|
size--;
|
|
dest++;
|
|
readstatus:
|
|
if (size > 0)
|
|
{
|
|
rc = ioctl (fd, PPRSTATUS, &status);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n",
|
|
strerror (errno), __FILE__, __LINE__);
|
|
word = status & 0x10;
|
|
error = status & 0x08;
|
|
}
|
|
}
|
|
while ((size > 0) && (!error) && (!word));
|
|
}
|
|
while ((size < 4) && (!error) && (size > 0));
|
|
|
|
/* here size=0 or error=8 or word=0x10 */
|
|
if ((word) && (!error) && (size))
|
|
{
|
|
rc = read (fd, dest, 4);
|
|
dest += 4;
|
|
size -= 4;
|
|
if (size != 0)
|
|
error = 0x08;
|
|
}
|
|
if (!error)
|
|
{
|
|
c = 0;
|
|
rc = ioctl (fd, PPRSTATUS, &status);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
error = status & 0x08;
|
|
if (error)
|
|
c = WaitOnError ();
|
|
}
|
|
else
|
|
{ /* 8282 */
|
|
c = WaitOnError ();
|
|
if (c == 0)
|
|
goto readstatus;
|
|
}
|
|
if (c == 1)
|
|
{
|
|
bread -= size;
|
|
}
|
|
else
|
|
{
|
|
bread += 3;
|
|
size = 3;
|
|
do
|
|
{
|
|
do
|
|
{
|
|
rc = read (fd, dest, 1);
|
|
dest++;
|
|
size--;
|
|
if (size)
|
|
{
|
|
rc = ioctl (fd, PPRSTATUS, &status);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n",
|
|
strerror (errno), __FILE__, __LINE__);
|
|
error = status & 0x08;
|
|
if (!error)
|
|
{
|
|
rc = ioctl (fd, PPRSTATUS, &status);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n",
|
|
strerror (errno), __FILE__, __LINE__);
|
|
error = status & 0x08;
|
|
}
|
|
}
|
|
}
|
|
while ((size > 0) && (!error));
|
|
c = 0;
|
|
if (error)
|
|
c = WaitOnError ();
|
|
}
|
|
while ((size > 0) && (c == 0));
|
|
}
|
|
|
|
/* end reading */
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
bval = 0xA0;
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = write (fd, &bval, 1);
|
|
|
|
mode = 1; /* data_reverse */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
mode = IEEE1284_MODE_EPP | IEEE1284_DATA;
|
|
rc = ioctl (fd, PPSETMODE, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
rc = read (fd, dest, 1);
|
|
bread++;
|
|
|
|
mode = 0; /* forward */
|
|
rc = ioctl (fd, PPDATADIR, &mode);
|
|
if (rc)
|
|
DBG (0, "ppdev ioctl returned <%s> (%s:%d)\n", strerror (errno),
|
|
__FILE__, __LINE__);
|
|
return (bread);
|
|
}
|
|
#endif
|
|
|
|
|
|
/* read up to size bytes, returns bytes read */
|
|
static int
|
|
DirectPausedReadBuffer (int size, unsigned char *dest)
|
|
{
|
|
int control;
|
|
int status;
|
|
int error;
|
|
int word;
|
|
int read;
|
|
int c;
|
|
|
|
/* init */
|
|
read = 0;
|
|
error = 0;
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control | 0x20);
|
|
if ((size & 0x03) != 0)
|
|
{
|
|
/* 8174 */
|
|
while ((!error) && ((size & 0x03) != 0))
|
|
{
|
|
Insb (EPPDATA, dest, 1);
|
|
size--;
|
|
dest++;
|
|
read++;
|
|
status = Inb (STATUS) & 0x1F;
|
|
error = status & 0x08;
|
|
}
|
|
if (error)
|
|
{
|
|
DBG (0, "Read error (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
/* from here, we read 1 byte, then size/4-1 32 bits words, and then
|
|
3 bytes, pausing on ERROR bit of STATUS */
|
|
size -= 4;
|
|
|
|
/* sanity test, seems to be wrongly handled ... */
|
|
if (size == 0)
|
|
{
|
|
DBG (0, "case not handled! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
word = 0;
|
|
error = 0;
|
|
read += size;
|
|
do
|
|
{
|
|
do
|
|
{
|
|
Insb (EPPDATA, dest, 1);
|
|
size--;
|
|
dest++;
|
|
readstatus:
|
|
if (size > 0)
|
|
{
|
|
status = Inb (STATUS) & 0x1F;
|
|
word = status & 0x10;
|
|
error = status & 0x08;
|
|
}
|
|
}
|
|
while ((size > 0) && (!error) && (!word));
|
|
}
|
|
while ((size < 4) && (!error) && (size > 0));
|
|
|
|
/* here size=0 or error=8 or word=0x10 */
|
|
if ((word) && (!error) && (size))
|
|
{
|
|
if (epp32)
|
|
Insw (EPPDATA, dest, 1);
|
|
else
|
|
Insb (EPPDATA, dest, 4);
|
|
dest += 4;
|
|
size -= 4;
|
|
if (size != 0)
|
|
error = 0x08;
|
|
}
|
|
if (!error)
|
|
{
|
|
c = 0;
|
|
error = Inb (STATUS) & 0x08;
|
|
if (error)
|
|
c = WaitOnError ();
|
|
}
|
|
else
|
|
{ /* 8282 */
|
|
c = WaitOnError ();
|
|
if (c == 0)
|
|
goto readstatus;
|
|
}
|
|
if (c == 1)
|
|
{
|
|
read -= size;
|
|
}
|
|
else
|
|
{
|
|
read += 3;
|
|
size = 3;
|
|
do
|
|
{
|
|
do
|
|
{
|
|
Insb (EPPDATA, dest, 1);
|
|
dest++;
|
|
size--;
|
|
if (size)
|
|
{
|
|
error = Inb (STATUS) & 0x08;
|
|
if (!error)
|
|
error = Inb (STATUS) & 0x08;
|
|
}
|
|
}
|
|
while ((size > 0) && (!error));
|
|
c = 0;
|
|
if (error)
|
|
c = WaitOnError ();
|
|
}
|
|
while ((size > 0) && (c == 0));
|
|
}
|
|
|
|
/* end reading */
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control);
|
|
EPPBlockMode (0xA0);
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control | 0x20);
|
|
Insb (EPPDATA, dest, 1);
|
|
read++;
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control);
|
|
return (read);
|
|
}
|
|
|
|
|
|
int
|
|
PausedReadBuffer (int size, unsigned char *dest)
|
|
{
|
|
#ifdef HAVE_LINUX_PPDEV_H
|
|
if (sanei_umax_pp_getparport () > 0)
|
|
return (ParportPausedReadBuffer (size, dest));
|
|
#endif
|
|
return (DirectPausedReadBuffer (size, dest));
|
|
}
|
|
|
|
|
|
|
|
|
|
/* returns 1 on success, 0 otherwise */
|
|
static int
|
|
SendWord1220P (int *cmd)
|
|
{
|
|
int i;
|
|
int reg;
|
|
int try = 0;
|
|
|
|
/* send header */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
retry:
|
|
RegisterWrite (0x1C, 0x55);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
RegisterWrite (0x1C, 0xAA);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* sync when needed */
|
|
if ((reg & 0x08) == 0x00)
|
|
{
|
|
reg = RegisterRead (0x1C);
|
|
DBG (16, "UTA: reg1C=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
if (((reg & 0x10) != 0x10) && (reg != 0x6B) && (reg != 0xAB)
|
|
&& (reg != 0x23))
|
|
{
|
|
DBG (0, "SendWord failed (reg1C=0x%02X) (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
usleep (1000);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if (reg != 0xC8)
|
|
{
|
|
DBG (0, "Unexpected reg19=0x%2X (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
}
|
|
}
|
|
do
|
|
{
|
|
if ((reg != 0xC0) && (reg != 0xC8))
|
|
{
|
|
DBG (0, "Unexpected reg19=0x%2X (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
}
|
|
/* 0xF0 certainly means error */
|
|
if ((reg == 0xC0) || (reg == 0xD0))
|
|
{
|
|
try++;
|
|
goto retry;
|
|
}
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
while (reg != 0xC8);
|
|
}
|
|
|
|
/* send word */
|
|
i = 0;
|
|
while ((reg == 0xC8) && (cmd[i] != -1))
|
|
{
|
|
RegisterWrite (0x1C, cmd[i]);
|
|
i++;
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
TRACE (16, "SendWord() passed ");
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "SendWord failed got 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on .....\n");
|
|
}
|
|
if (((reg == 0xC0) || (reg == 0xD0)) && (cmd[i] != -1))
|
|
{
|
|
DBG (0, "SendWord failed: short send (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
reg = RegisterRead (0x1C);
|
|
DBG (16, "SendWord, reg1C=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
/* model 0x07 has always the last bit set to 1, and even bit 1 */
|
|
/* when UTA is present, we get 0x6B there */
|
|
scannerStatus = reg & 0xFC;
|
|
if (scannerStatus == 0x68)
|
|
hasUTA = 1;
|
|
|
|
reg = reg & 0x10;
|
|
if ((reg != 0x10) && (scannerStatus != 0x68) && (scannerStatus != 0xA8))
|
|
{
|
|
DBG (0, "SendWord failed: acknowledge not received (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
if (try)
|
|
{
|
|
DBG (0, "SendWord retry success (retry %d time%s) ... (%s:%d)\n", try,
|
|
(try > 1) ? "s" : "", __FILE__, __LINE__);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* returns 1 on success, 0 otherwise */
|
|
static int
|
|
SendWord610P (int *cmd)
|
|
{
|
|
int i;
|
|
int tmp;
|
|
|
|
Outb (CONTROL, 0x05);
|
|
tmp = Inb (CONTROL);
|
|
Outb (CONTROL, 0x04);
|
|
|
|
/* send magic tag */
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0xC8)
|
|
{
|
|
DBG (0,
|
|
"SendWord610P failed, expected tmp=0xC8 , found 0x%02X (%s:%d)\n",
|
|
tmp, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
tmp = Inb (CONTROL);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (EPPDATA, 0x55);
|
|
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0xC8)
|
|
{
|
|
DBG (0,
|
|
"SendWord610P failed, expected tmp=0xC8 , found 0x%02X (%s:%d)\n",
|
|
tmp, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
tmp = Inb (CONTROL);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (EPPDATA, 0xAA);
|
|
|
|
tmp = Inb (CONTROL);
|
|
Outb (CONTROL, 0xA4);
|
|
for (i = 0; i < 9; i++)
|
|
{
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0xC8)
|
|
{
|
|
DBG (0,
|
|
"SendWord610P failed, expected tmp=0xC8 , found 0x%02X (%s:%d)\n",
|
|
tmp, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
i = 0;
|
|
while ((tmp == 0xC8) && (cmd[i] != -1))
|
|
{
|
|
tmp = Inb (STATUS);
|
|
Inb (CONTROL);
|
|
Outb (CONTROL, 0x04);
|
|
Outb (EPPDATA, cmd[i]);
|
|
}
|
|
|
|
/* end */
|
|
Outb (DATA, 0xFF);
|
|
tmp = Inb (CONTROL);
|
|
Outb (CONTROL, 0xA4);
|
|
tmp = Inb (STATUS);
|
|
if ((tmp != 0xC0) && (tmp != 0xD0))
|
|
{
|
|
DBG (0,
|
|
"SendWord610P failed got 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
tmp, __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on .....\n");
|
|
}
|
|
tmp = Inb (EPPDATA);
|
|
scannerStatus = tmp;
|
|
return (1);
|
|
}
|
|
|
|
/*
|
|
* 0: failure
|
|
* 1: success
|
|
*/
|
|
static int
|
|
SendWord (int *cmd)
|
|
{
|
|
switch (sanei_umax_pp_getastra ())
|
|
{
|
|
case 610:
|
|
return (SendWord610P (cmd));
|
|
case 1220:
|
|
case 1600:
|
|
case 2000:
|
|
default:
|
|
return (SendWord1220P (cmd));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
/* RingScanner: returns 1 if scanner present, else 0 */
|
|
/******************************************************************************/
|
|
/*
|
|
* in fact this function is really close to CPP macro in
|
|
* /usr/src/linux/drivers/block/paride/epat.c .....
|
|
* we have almost CPP(8)
|
|
*/
|
|
|
|
static int
|
|
RingScanner (int count, unsigned long delay)
|
|
{
|
|
int status;
|
|
int data;
|
|
int control;
|
|
int ret = 1;
|
|
|
|
/* save state */
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x1F;
|
|
|
|
/* send -irq,+reset */
|
|
Outb (CONTROL, (control & 0xF) | 0x4);
|
|
|
|
/* unhandled case */
|
|
if (g674 == 1)
|
|
{
|
|
DBG (1, "OUCH! %s:%d\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* send ring string */
|
|
Outb (DATA, 0x22);
|
|
usleep (delay);
|
|
Outb (DATA, 0x22);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x22);
|
|
usleep (delay);
|
|
Outb (DATA, 0x22);
|
|
usleep (delay);
|
|
Outb (DATA, 0x22);
|
|
usleep (delay);
|
|
}
|
|
Outb (DATA, 0xAA);
|
|
usleep (delay);
|
|
Outb (DATA, 0xAA);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0xAA);
|
|
usleep (delay);
|
|
Outb (DATA, 0xAA);
|
|
usleep (delay);
|
|
Outb (DATA, 0xAA);
|
|
usleep (delay);
|
|
}
|
|
Outb (DATA, 0x55);
|
|
usleep (delay);
|
|
Outb (DATA, 0x55);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x55);
|
|
usleep (delay);
|
|
Outb (DATA, 0x55);
|
|
usleep (delay);
|
|
Outb (DATA, 0x55);
|
|
usleep (delay);
|
|
}
|
|
Outb (DATA, 0x00);
|
|
usleep (delay);
|
|
Outb (DATA, 0x00);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x00);
|
|
usleep (delay);
|
|
Outb (DATA, 0x00);
|
|
usleep (delay);
|
|
Outb (DATA, 0x00);
|
|
usleep (delay);
|
|
}
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
}
|
|
|
|
/* OK ? */
|
|
status = Inb (STATUS);
|
|
usleep (delay);
|
|
if ((status & 0xB8) != 0xB8)
|
|
{
|
|
DBG (1, "status %d doesn't match! %s:%d\n", status, __FILE__, __LINE__);
|
|
ret = 0;
|
|
}
|
|
|
|
/* if OK send 0x87 */
|
|
if (ret)
|
|
{
|
|
Outb (DATA, 0x87);
|
|
usleep (delay);
|
|
Outb (DATA, 0x87);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x87);
|
|
usleep (delay);
|
|
Outb (DATA, 0x87);
|
|
usleep (delay);
|
|
Outb (DATA, 0x87);
|
|
usleep (delay);
|
|
}
|
|
status = Inb (STATUS);
|
|
/* status = 126 when scanner not connected .... */
|
|
if ((status & 0xB8) != 0x18)
|
|
{
|
|
DBG (1, "status %d doesn't match! %s:%d\n", status, __FILE__,
|
|
__LINE__);
|
|
ret = 0;
|
|
}
|
|
}
|
|
|
|
/* if OK send 0x78 */
|
|
if (ret)
|
|
{
|
|
Outb (DATA, 0x78);
|
|
usleep (delay);
|
|
Outb (DATA, 0x78);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x78);
|
|
usleep (delay);
|
|
Outb (DATA, 0x78);
|
|
usleep (delay);
|
|
Outb (DATA, 0x78);
|
|
usleep (delay);
|
|
}
|
|
status = Inb (STATUS);
|
|
if ((status & 0x30) != 0x30)
|
|
{
|
|
DBG (1, "status %d doesn't match! %s:%d\n", status, __FILE__,
|
|
__LINE__);
|
|
ret = 0;
|
|
}
|
|
}
|
|
|
|
/* ring OK, send termination */
|
|
if (ret)
|
|
{
|
|
Outb (DATA, 0x08);
|
|
usleep (delay);
|
|
Outb (DATA, 0x08);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0x08);
|
|
usleep (delay);
|
|
Outb (DATA, 0x08);
|
|
usleep (delay);
|
|
Outb (DATA, 0x08);
|
|
usleep (delay);
|
|
}
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
if (count == 5)
|
|
{
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
Outb (DATA, 0xFF);
|
|
usleep (delay);
|
|
}
|
|
}
|
|
|
|
/* restore state */
|
|
Outb (CONTROL, control);
|
|
Outb (DATA, data);
|
|
return (ret);
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/* test some version : returns 1 on success, 0 otherwise */
|
|
/*****************************************************************************/
|
|
|
|
|
|
static int
|
|
TestVersion (int no)
|
|
{
|
|
int data;
|
|
int status;
|
|
int control;
|
|
int count;
|
|
int tmp;
|
|
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x3F;
|
|
Outb (CONTROL, (control & 0x1F) | 0x04);
|
|
|
|
/* send magic sequence */
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0x22);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x55);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0x00);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0xFF);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x87);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
Outb (DATA, 0x78);
|
|
tmp = no | 0x88;
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
|
|
/* test status */
|
|
status = Inb (STATUS);
|
|
status = Inb (STATUS);
|
|
if ((status & 0xB8) != 0)
|
|
{
|
|
/* 1600P fails here */
|
|
DBG (64, "status %d doesn't match! %s:%d\n", status, __FILE__,
|
|
__LINE__);
|
|
Outb (CONTROL, control);
|
|
Outb (DATA, data);
|
|
return 0;
|
|
}
|
|
|
|
count = 0xF0;
|
|
do
|
|
{
|
|
tmp = no | 0x80;
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
tmp = no | 0x88;
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
Outb (DATA, tmp);
|
|
|
|
/* command received ? */
|
|
status = Inb (STATUS);
|
|
status = ((status << 1) & 0x70) | (status & 0x80);
|
|
if (status != count)
|
|
{
|
|
/* since failure is expected, we dont't alaways print */
|
|
/* this message ... */
|
|
DBG (2, "status %d doesn't match count 0x%X! %s:%d\n", status,
|
|
count, __FILE__, __LINE__);
|
|
Outb (CONTROL, control);
|
|
Outb (DATA, data);
|
|
return (0);
|
|
}
|
|
|
|
/* next */
|
|
count -= 0x10;
|
|
}
|
|
while (count > 0);
|
|
|
|
/* restore port , successful exit */
|
|
Outb (CONTROL, control);
|
|
Outb (DATA, data);
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* sends len bytes to scanner */
|
|
/* needs data channel to be set up */
|
|
/* returns 1 on success, 0 otherwise */
|
|
static int
|
|
SendLength (int *cmd, int len)
|
|
{
|
|
int i;
|
|
int reg, wait;
|
|
int try = 0;
|
|
|
|
/* send header */
|
|
retry:
|
|
wait = RegisterRead (0x19) & 0xF8;
|
|
|
|
RegisterWrite (0x1C, 0x55);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
RegisterWrite (0x1C, 0xAA);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* sync when needed */
|
|
if ((wait & 0x08) == 0x00)
|
|
{
|
|
reg = RegisterRead (0x1C);
|
|
while (((reg & 0x10) != 0x10) && (reg != 0x6B) && (reg != 0xAB)
|
|
&& (reg != 0x23))
|
|
{
|
|
DBG (0,
|
|
"SendLength failed, expected reg & 0x10=0x10 , found 0x%02X (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
if (try > 10)
|
|
{
|
|
DBG (0, "Aborting...\n");
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "Retrying ...\n");
|
|
}
|
|
/* resend */
|
|
Epilogue ();
|
|
Prologue ();
|
|
try++;
|
|
goto retry;
|
|
}
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if (reg != 0xC8)
|
|
{
|
|
DBG (0, "Unexpected reg19=0x%2X (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
/* 0xF0 certainly means error */
|
|
if ((reg == 0xC0) || (reg == 0xD0) || (reg == 0x80))
|
|
{
|
|
/* resend */
|
|
try++;
|
|
if (try > 20)
|
|
{
|
|
DBG (0, "SendLength retry failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
|
|
Epilogue ();
|
|
SendCommand (0x00);
|
|
SendCommand (0xE0);
|
|
Outb (DATA, 0x00);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x04);
|
|
SendCommand (0x30);
|
|
|
|
Prologue ();
|
|
goto retry;
|
|
}
|
|
}
|
|
}
|
|
do
|
|
{
|
|
if ((reg != 0xC0) && (reg != 0xD0) && (reg != 0xC8))
|
|
{
|
|
/* status has changed while waiting */
|
|
/* but it's too early */
|
|
DBG (0, "Unexpected reg19=0x%2X (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
}
|
|
/* 0xF0 certainly means error */
|
|
if ((reg == 0xC0) || (reg == 0xD0) || (reg == 0x80))
|
|
{
|
|
/* resend */
|
|
try++;
|
|
Epilogue ();
|
|
|
|
SendCommand (0x00);
|
|
SendCommand (0xE0);
|
|
Outb (DATA, 0x00);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x04);
|
|
SendCommand (0x30);
|
|
|
|
Prologue ();
|
|
goto retry;
|
|
}
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
while (reg != 0xC8);
|
|
}
|
|
|
|
/* send bytes */
|
|
i = 0;
|
|
while ((reg == 0xC8) && (i < len))
|
|
{
|
|
/* write byte */
|
|
RegisterWrite (0x1C, cmd[i]);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* 1B handling: escape it to confirm value */
|
|
if (cmd[i] == 0x1B)
|
|
{
|
|
RegisterWrite (0x1C, cmd[i]);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
i++;
|
|
}
|
|
DBG (16, "SendLength, reg19=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0,
|
|
"SendLength failed got 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on .....\n");
|
|
}
|
|
|
|
/* check if 'finished status' received too early */
|
|
if (((reg == 0xC0) || (reg == 0xD0)) && (i != len))
|
|
{
|
|
DBG (0, "SendLength failed: sent only %d bytes out of %d (%s:%d)\n", i,
|
|
len, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
|
|
reg = RegisterRead (0x1C);
|
|
DBG (16, "SendLength, reg1C=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
|
|
/* model 0x07 has always the last bit set to 1 */
|
|
scannerStatus = reg & 0xFC;
|
|
reg = reg & 0x10;
|
|
if ((reg != 0x10) && (scannerStatus != 0x68) && (scannerStatus != 0xA8))
|
|
{
|
|
DBG (0, "SendLength failed: acknowledge not received (%s:%d)\n",
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
if (try)
|
|
{
|
|
DBG (0, "SendLength retry success (retry %d time%s) ... (%s:%d)\n", try,
|
|
(try > 1) ? "s" : "", __FILE__, __LINE__);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* sends data bytes to scanner */
|
|
/* needs data channel to be set up */
|
|
/* returns 1 on success, 0 otherwise */
|
|
static int
|
|
SendData (int *cmd, int len)
|
|
{
|
|
int i;
|
|
int reg;
|
|
|
|
/* send header */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* send bytes */
|
|
i = 0;
|
|
while ((reg == 0xC8) && (i < len))
|
|
{
|
|
/* write byte */
|
|
RegisterWrite (0x1C, cmd[i]);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* 1B handling: escape it to confirm value */
|
|
if (cmd[i] == 0x1B)
|
|
{
|
|
RegisterWrite (0x1C, 0x1B);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
|
|
/* escape 55 AA pattern by adding 1B */
|
|
if ((i < len - 1) && (cmd[i] == 0x55) && (cmd[i + 1] == 0xAA))
|
|
{
|
|
RegisterWrite (0x1C, 0x1B);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
|
|
/* next value */
|
|
i++;
|
|
}
|
|
DBG (16, "SendData, reg19=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "SendData failed got 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on .....\n");
|
|
}
|
|
|
|
/* check if 'finished status' received too early */
|
|
if (((reg == 0xC0) || (reg == 0xD0)) && (i < len))
|
|
{
|
|
DBG (0, "SendData failed: sent only %d bytes out of %d (%s:%d)\n", i,
|
|
len, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
|
|
reg = RegisterRead (0x1C);
|
|
DBG (16, "SendData, reg1C=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
|
|
/* model 0x07 has always the last bit set to 1 */
|
|
scannerStatus = reg & 0xFC;
|
|
reg = reg & 0x10;
|
|
if ((reg != 0x10) && (scannerStatus != 0x68) && (scannerStatus != 0xA8)
|
|
&& (scannerStatus != 0x20))
|
|
{
|
|
DBG (0, "SendData failed: acknowledge not received (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* receive data bytes from scanner */
|
|
/* needs data channel to be set up */
|
|
/* returns 1 on success, 0 otherwise */
|
|
/* uses PausedReadBuffer */
|
|
static int
|
|
PausedReadData (int size, unsigned char *dest)
|
|
{
|
|
int reg;
|
|
int tmp;
|
|
int read;
|
|
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "Unexpected reg19: 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
REGISTERREAD (0x0C, 0x04);
|
|
REGISTERWRITE (0x0C, 0x44); /* sets data direction ? */
|
|
EPPBlockMode (0x80);
|
|
read = PausedReadBuffer (size, dest);
|
|
if (read < size)
|
|
{
|
|
DBG (16,
|
|
"PausedReadBuffer(%d,dest) failed, only got %d bytes (%s:%d)\n",
|
|
size, read, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "PausedReadBuffer(%d,dest) passed (%s:%d)\n", size, __FILE__,
|
|
__LINE__);
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
/* receive data bytes from scanner */
|
|
/* needs data channel to be set up */
|
|
/* returns 1 on success, 0 otherwise */
|
|
static int
|
|
ReceiveData (int *cmd, int len)
|
|
{
|
|
int i;
|
|
int reg;
|
|
|
|
/* send header */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* send bytes */
|
|
i = 0;
|
|
while (((reg == 0xD0) || (reg == 0xC0)) && (i < len))
|
|
{
|
|
/* write byte */
|
|
cmd[i] = RegisterRead (0x1C);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
i++;
|
|
}
|
|
DBG (16, "ReceiveData, reg19=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "SendData failed got 0x%02X instead of 0xC0 or 0xD0 (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on .....\n");
|
|
}
|
|
|
|
/* check if 'finished status' received to early */
|
|
if (((reg == 0xC0) || (reg == 0xD0)) && (i != len))
|
|
{
|
|
DBG (0,
|
|
"ReceiveData failed: received only %d bytes out of %d (%s:%d)\n",
|
|
i, len, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
|
|
reg = RegisterRead (0x1C);
|
|
DBG (16, "ReceiveData, reg1C=0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
|
|
/* model 0x07 has always the last bit set to 1 */
|
|
scannerStatus = reg & 0xF8;
|
|
reg = reg & 0x10;
|
|
if ((reg != 0x10) && (scannerStatus != 0x68) && (scannerStatus != 0xA8))
|
|
{
|
|
DBG (0, "ReceiveData failed: acknowledge not received (%s:%d)\n",
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* 1=success, 0 failed */
|
|
static int
|
|
Fonc001 (void)
|
|
{
|
|
int i;
|
|
int res;
|
|
int reg;
|
|
|
|
res = 1;
|
|
while (res == 1)
|
|
{
|
|
RegisterWrite (0x1A, 0x0C);
|
|
RegisterWrite (0x18, 0x40);
|
|
|
|
/* send 0x06 */
|
|
RegisterWrite (0x1A, 0x06);
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg & 0x78) == 0x38)
|
|
{
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (res == 1)
|
|
{
|
|
RegisterWrite (0x1A, 0x00);
|
|
RegisterWrite (0x1A, 0x0C);
|
|
}
|
|
}
|
|
|
|
/* send 0x07 */
|
|
RegisterWrite (0x1A, 0x07);
|
|
res = 1;
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg & 0x78) == 0x38)
|
|
{
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (res != 0)
|
|
return (0);
|
|
|
|
/* send 0x04 */
|
|
RegisterWrite (0x1A, 0x04);
|
|
res = 1;
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg & 0xF8) == 0xF8)
|
|
{
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (res != 0)
|
|
return (0);
|
|
|
|
/* send 0x05 */
|
|
RegisterWrite (0x1A, 0x05);
|
|
res = 1;
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
reg = RegisterRead (0x1A);
|
|
if (reg == 0x05)
|
|
{
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (res != 0)
|
|
return (0);
|
|
|
|
/* end */
|
|
RegisterWrite (0x1A, 0x84);
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 1 OK, 0 failed */
|
|
static int
|
|
FoncSendWord (int *cmd)
|
|
{
|
|
int reg, tmp;
|
|
|
|
Init022 ();
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
/* return -1 */
|
|
DBG (0, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n", gEPAT,
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
DBG (16, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n", reg,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
if (SendWord (cmd) == 0)
|
|
{
|
|
DBG (0, "SendWord(cmd) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
|
|
/* termination sequence */
|
|
REGISTERWRITE (0x0A, 0x00);
|
|
REGISTERREAD (0x0D, 0x40);
|
|
REGISTERWRITE (0x0D, 0x00);
|
|
if (GetModel () != 0x07)
|
|
SendCommand (40);
|
|
SendCommand (30);
|
|
Outb (DATA, 0x04);
|
|
tmp = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, tmp);
|
|
return (1);
|
|
}
|
|
|
|
|
|
static int
|
|
CmdSetDataBuffer (int *data)
|
|
{
|
|
int cmd1[] = { 0x00, 0x00, 0x22, 0x88, -1 }; /* 34 bytes write on channel 8 */
|
|
int cmd2[] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x03, 0xC1, 0x80,
|
|
0x00, 0x20, 0x02, 0x00, 0x16, 0x41, 0xE0, 0xAC, 0x03, 0x03, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, -1
|
|
};
|
|
int cmd3[] = { 0x00, 0x08, 0x00, 0x84, -1 }; /* 2048 bytes size write on channel 4 (data) */
|
|
int cmd4[] = { 0x00, 0x08, 0x00, 0xC4, -1 }; /* 2048 bytes size read on channel 4 (data) */
|
|
int i;
|
|
unsigned char dest[2048];
|
|
|
|
/* CmdSet(8,34,cmd2), but without prologue/epilogue */
|
|
/* set block length to 34 bytes on 'channel 8' */
|
|
SendWord (cmd1);
|
|
DBG (16, "SendWord(cmd1) passed (%s:%d) \n", __FILE__, __LINE__);
|
|
|
|
/* SendData */
|
|
SendData (cmd2, 0x22);
|
|
DBG (16, "SendData(cmd2) passed (%s:%d) \n", __FILE__, __LINE__);
|
|
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Bloc8Decode (cmd2);
|
|
}
|
|
|
|
/* set block length to 2048, write on 'channel 4' */
|
|
SendWord (cmd3);
|
|
DBG (16, "SendWord(cmd3) passed (%s:%d) \n", __FILE__, __LINE__);
|
|
|
|
if (SendData (data, 2048) == 0)
|
|
{
|
|
DBG (0, "SendData(data,%d) failed (%s:%d)\n", 2048, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendData(data,2048) passed ...");
|
|
|
|
/* read back all data sent to 'channel 4' */
|
|
SendWord (cmd4);
|
|
DBG (16, "SendWord(cmd4) passed (%s:%d) \n", __FILE__, __LINE__);
|
|
|
|
if (PausedReadData (2048, dest) == 0)
|
|
{
|
|
DBG (16, "PausedReadData(2048,dest) failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "PausedReadData(2048,dest) passed (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
/* dest should hold the same datas than donnees */
|
|
for (i = 0; i < 2047; i++)
|
|
{
|
|
if (data[i] != (int) (dest[i]))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Warning data read back differs: expected %02X found dest[%d]=%02X ! (%s:%d)\n",
|
|
data[i], i, dest[i], __FILE__, __LINE__);
|
|
}
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
/* free scanner and parallel port
|
|
0: failure
|
|
1: success
|
|
*/
|
|
int
|
|
sanei_umax_pp_ReleaseScanner (void)
|
|
{
|
|
int reg;
|
|
|
|
REGISTERWRITE (0x0A, 0x00);
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xBF);
|
|
RegisterWrite (0x0D, reg);
|
|
if (GetModel () != 0x07)
|
|
{
|
|
if (SendCommand (0x40) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0x40) (%s:%d) failed ...\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
if (SendCommand (0x30) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0x30) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (1, "ReleaseScanner() done ...\n");
|
|
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
/* 1: OK
|
|
0: end session failed */
|
|
|
|
int
|
|
sanei_umax_pp_EndSession (void)
|
|
{
|
|
int tmp;
|
|
int control;
|
|
int data;
|
|
int reg;
|
|
int zero[5] = { 0, 0, 0, 0, -1 };
|
|
|
|
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control);
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control);
|
|
|
|
g67D = 1;
|
|
|
|
if (SendCommand (0xE0) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0xE0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0xE0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
g6FE = 1;
|
|
g674 = 0;
|
|
ClearRegister (0);
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
REGISTERREAD (0x0B, 0xC7);
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg | 0x43);
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
if (reg != 0x1C)
|
|
DBG (0, "Expected 0x00 found 0x%02X .... (%s:%d)\n", reg, __FILE__,
|
|
__LINE__);
|
|
else
|
|
{
|
|
DBG (16, "Previous probe detected .... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
}
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
|
|
if (SendWord (zero) == 0)
|
|
{
|
|
DBG (16, "SendWord(zero) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
Epilogue ();
|
|
|
|
sanei_umax_pp_CmdSync (0xC2);
|
|
sanei_umax_pp_CmdSync (0x00); /* cancels any pending operation */
|
|
sanei_umax_pp_CmdSync (0x00); /* cancels any pending operation */
|
|
sanei_umax_pp_ReleaseScanner ();
|
|
|
|
/* restore port state */
|
|
Outb (DATA, 0x04);
|
|
Outb (CONTROL, gControl);
|
|
|
|
/* OUF */
|
|
DBG (1, "End session done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* 1: OK
|
|
2: homing happened
|
|
3: scanner busy
|
|
0: init failed
|
|
|
|
init transport layer
|
|
init scanner
|
|
*/
|
|
|
|
int
|
|
sanei_umax_pp_InitScanner (int recover)
|
|
{
|
|
int i, j;
|
|
int status;
|
|
int readcmd[64];
|
|
int sentcmd[64];
|
|
|
|
|
|
|
|
|
|
|
|
/* in umax1220u, this buffer is opc[16] */
|
|
j = 0;
|
|
sentcmd[j] = 0x02;
|
|
j++;
|
|
sentcmd[j] = 0x80;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x70;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x2F;
|
|
j++;
|
|
sentcmd[j] = 0x2F;
|
|
j++;
|
|
sentcmd[j] = 0x07;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
sentcmd[j] = 0x80;
|
|
j++;
|
|
sentcmd[j] = 0xF0;
|
|
j++; /* F0 means brightness on, 90 brightness off */
|
|
if (GetModel () == 0x07)
|
|
{
|
|
sentcmd[j] = 0x00;
|
|
j++;
|
|
}
|
|
else
|
|
{
|
|
sentcmd[j] = 0x18;
|
|
j++;
|
|
}
|
|
sentcmd[j] = -1;
|
|
/* fails here if there is an unfinished previous scan */
|
|
if (CmdSetGet (0x02, j, sentcmd) != 1)
|
|
{
|
|
DBG (0, "CmdSetGet(0x02,j,sentcmd) failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetGet(0x02,j,sentcmd) passed ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
/* needs some init */
|
|
if (sentcmd[15] == 0x18)
|
|
{
|
|
sentcmd[15] = 0x00; /* was 0x18 */
|
|
if (CmdSetGet (0x02, j, sentcmd) != 1)
|
|
{
|
|
DBG (0, "CmdSetGet(0x02,j,sentcmd) failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetGet(0x02,j,sentcmd) passed ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
/* in umax1220u, this buffer does not exist */
|
|
j = 0;
|
|
sentcmd[j] = 0xA0;
|
|
j++;
|
|
sentcmd[j] = 0xA1;
|
|
j++;
|
|
sentcmd[j] = 0xA2;
|
|
j++;
|
|
sentcmd[j] = 0xA3;
|
|
j++;
|
|
sentcmd[j] = 0xA4;
|
|
j++;
|
|
sentcmd[j] = 0xA5;
|
|
j++;
|
|
sentcmd[j] = 0xA6;
|
|
j++;
|
|
sentcmd[j] = 0xA7;
|
|
j++;
|
|
sentcmd[j] = -1;
|
|
if (CmdSetGet (0x01, j, sentcmd) != 1)
|
|
{
|
|
DBG (0, "CmdSetGet(0x02,j,sentcmd) failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetGet(0x01,j,sentcmd) passed ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
|
|
/* ~ opb3: inquire status */
|
|
if (CmdGet (0x08, 36, readcmd) == 0)
|
|
{
|
|
DBG (0, "CmdGet(0x08,36,readcmd) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
if (DBG_LEVEL >= 32)
|
|
{
|
|
Bloc8Decode (readcmd);
|
|
}
|
|
DBG (16, "CmdGet(0x08,36,readcmd) passed (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
/* is the scanner busy parking ? */
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
DBG (8, "INQUIRE SCANNER STATUS IS 0x%02X (%s:%d)\n", status, __FILE__,
|
|
__LINE__);
|
|
if ((!recover) && (status & MOTOR_BIT) == 0x00)
|
|
{
|
|
DBG (1, "Warning: scanner motor on, giving up ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (3);
|
|
}
|
|
|
|
/* head homing needed ? */
|
|
if ((readcmd[34] != 0x1A) || (recover == 1))
|
|
{ /* homing needed, readcmd[34] should be 0x48 */
|
|
int op01[17] =
|
|
{ 0x01, 0x00, 0x32, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x17, 0x05, 0x00,
|
|
0x00, 0x00, 0x80, 0xE4, 0x00, -1
|
|
};
|
|
int op05[17] =
|
|
{ 0x01, 0x00, 0x01, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x13, 0x05, 0x00,
|
|
0x00, 0x00, 0x80, 0xF0, 0x00, -1
|
|
};
|
|
int op02[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40,
|
|
0x01, 0x00, 0x20, 0x02, 0x00, 0x16, 0x00, 0x70, 0x9F, 0x06, 0x00,
|
|
0x00, 0xF6, 0x4D, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF,
|
|
0x0B, 0x1A, 0x00, -1
|
|
};
|
|
int op04[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x03, 0xC1,
|
|
0x80, 0x00, 0x20, 0x02, 0x00, 0x16, 0x80, 0x15, 0x78, 0x03, 0x03,
|
|
0x00, 0x00, 0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF,
|
|
0x0B, 0x1A, 0x00, -1
|
|
};
|
|
int op03[9] = { 0x00, 0x00, 0x00, 0xAA, 0xCC, 0xEE, 0xFF, 0xFF, -1 };
|
|
|
|
CMDSYNC (0xC2);
|
|
CMDSETGET (0x02, 16, op01);
|
|
CMDSETGET (0x08, 36, op02);
|
|
|
|
if (!recover)
|
|
{
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (0x04, 8, op03);
|
|
CMDSYNC (0x40);
|
|
do
|
|
{
|
|
sleep (1);
|
|
CMDSYNC (0xC2);
|
|
}
|
|
while ((sanei_umax_pp_ScannerStatus () & 0x90) != 0x90);
|
|
|
|
op01[2] = 0x1E;
|
|
op01[9] = 0x01;
|
|
CMDSETGET (0x02, 16, op01);
|
|
CMDSETGET (0x08, 36, op02);
|
|
CMDSYNC (0x00);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (0x04, 8, op03);
|
|
|
|
CMDSYNC (0x40);
|
|
do
|
|
{
|
|
sleep (1);
|
|
CMDSYNC (0xC2);
|
|
}
|
|
while ((sanei_umax_pp_ScannerStatus () & 0x90) != 0x90);
|
|
CMDSYNC (0x00);
|
|
}
|
|
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
|
|
do
|
|
{
|
|
usleep (500000);
|
|
CMDSYNC (0xC2);
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
status = status & 0x10;
|
|
}
|
|
while (status != 0x10); /* was 0x90 */
|
|
CMDSETGET (0x02, 16, op05);
|
|
CMDSETGET (0x08, 36, op04);
|
|
CMDSYNC (0x40);
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
DBG (16, "loop %d passed, status=0x%02X (%s:%d)\n", i, status,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
|
|
|
|
/* get head back home ... */
|
|
do
|
|
{
|
|
i++;
|
|
do
|
|
{
|
|
usleep (500000);
|
|
CMDSYNC (0xC2);
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
status = status & 0x10;
|
|
}
|
|
while (status != 0x10); /* was 0x90 */
|
|
CMDSETGET (0x02, 16, op05);
|
|
CMDSETGET (0x08, 36, op04);
|
|
CMDSYNC (0x40);
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
DBG (16, "loop %d passed, status=0x%02X (%s:%d)\n", i, status,
|
|
__FILE__, __LINE__);
|
|
}
|
|
while ((status & MOTOR_BIT) == 0x00); /* 0xD0 when head is back home */
|
|
|
|
do
|
|
{
|
|
usleep (500000);
|
|
CMDSYNC (0xC2);
|
|
}
|
|
while ((sanei_umax_pp_ScannerStatus () & 0x90) != 0x90);
|
|
|
|
|
|
/* don't do automatic home sequence on recovery */
|
|
if (!recover)
|
|
{
|
|
CMDSYNC (0x00);
|
|
op01[2] = 0x1A;
|
|
op01[3] = 0x74; /* was 0x70 */
|
|
op01[9] = 0x05; /* initial value */
|
|
op01[14] = 0xF4; /* was 0xE4 */
|
|
CMDSETGET (0x02, 16, op01);
|
|
CMDSETGET (0x08, 36, op02);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (0x04, 8, op03);
|
|
CMDSYNC (0x40);
|
|
|
|
/* wait for automatic homing sequence */
|
|
/* to complete, thus avoiding */
|
|
/* scanning too early */
|
|
do
|
|
{
|
|
/* the sleep is here to prevent */
|
|
/* excessive CPU usage, can be */
|
|
/* removed, if we don't care */
|
|
sleep (3);
|
|
CMDSYNC (0xC2);
|
|
DBG (16, "PARKING polling status is 0x%02X (%s:%d)\n",
|
|
sanei_umax_pp_ScannerStatus (), __FILE__, __LINE__);
|
|
}
|
|
while (sanei_umax_pp_ScannerStatus () == 0x90);
|
|
}
|
|
|
|
/* signal homing */
|
|
return (2);
|
|
}
|
|
|
|
|
|
/* end ... */
|
|
DBG (1, "Scanner init done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
/*
|
|
1: OK
|
|
2: failed, try again
|
|
0: init failed
|
|
|
|
initialize the transport layer
|
|
|
|
*/
|
|
|
|
static int
|
|
InitTransport610P (int recover)
|
|
{
|
|
int control;
|
|
int data;
|
|
int zero[5] = { 0, 0, 0, 0, -1 };
|
|
|
|
|
|
/* recover is not used yet, but will */
|
|
recover = 0; /* quit compiler quiet .. */
|
|
|
|
/* set port state */
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x0C;
|
|
Outb (CONTROL, control);
|
|
control = Inb (CONTROL) & 0x0C;
|
|
Outb (CONTROL, control);
|
|
gControl = 0x0C;
|
|
|
|
g67D = 1;
|
|
if (SendCommand (0xE0) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0xE0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0xE0) passed...\n");
|
|
g6FE = 1;
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed...\n");
|
|
|
|
/* sync */
|
|
Prologue ();
|
|
if (SendWord (zero) == 0)
|
|
{
|
|
DBG (0, "SendWord(zero) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendWord(zero) passed (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
Epilogue ();
|
|
|
|
/* OK ! */
|
|
DBG (1, "InitTransport610P done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
/*
|
|
1: OK
|
|
2: failed, try again
|
|
0: init failed
|
|
|
|
initialize the transport layer
|
|
|
|
*/
|
|
|
|
static int
|
|
InitTransport1220P (int recover)
|
|
{
|
|
int i, j;
|
|
int control;
|
|
int data;
|
|
int reg;
|
|
unsigned char *dest = NULL;
|
|
int zero[5] = { 0, 0, 0, 0, -1 };
|
|
int model;
|
|
|
|
/* init cancel handling */
|
|
/* InitCancel(); */
|
|
|
|
/* recover is not used yet, but will */
|
|
recover = 0; /* quit compiler quiet .. */
|
|
|
|
/* set port state */
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x0C;
|
|
Outb (CONTROL, control);
|
|
control = Inb (CONTROL) & 0x0C;
|
|
Outb (CONTROL, control);
|
|
gControl = 0x0C;
|
|
|
|
g67D = 1;
|
|
if (SendCommand (0xE0) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0xE0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0xE0) passed...\n");
|
|
g6FE = 1;
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed...\n");
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
gEPAT = 0xC7;
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
DBG (16, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n",
|
|
gEPAT, reg, __FILE__, __LINE__);
|
|
DBG (16, "Scanner needs probing ... \n");
|
|
sanei_umax_pp_ReleaseScanner ();
|
|
if (sanei_umax_pp_ProbeScanner (recover) != 1)
|
|
{
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
sanei_umax_pp_ReleaseScanner ();
|
|
return (2); /* signals retry InitTransport() */
|
|
}
|
|
}
|
|
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
if (reg != 0x1C)
|
|
{
|
|
DBG (0, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n",
|
|
reg, __FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (16, "Scanner in idle state .... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
}
|
|
|
|
/* model detection: redone since we might not be probing each time ... */
|
|
/* write addr in 0x0E, read value at 0x0F */
|
|
REGISTERWRITE (0x0E, 0x01);
|
|
model = RegisterRead (0x0F);
|
|
SetModel (model);
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
REGISTERWRITE (0x0A, 0x11);
|
|
|
|
|
|
dest = (unsigned char *) (malloc (65536));
|
|
if (dest == NULL)
|
|
{
|
|
DBG (0, "Failed to allocate 64 Ko !\n");
|
|
return (0);
|
|
}
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
dest[i * 2] = i;
|
|
dest[i * 2 + 1] = 0xFF - i;
|
|
dest[512 + i * 2] = i;
|
|
dest[512 + i * 2 + 1] = 0xFF - i;
|
|
}
|
|
for (i = 0; i < 150; i++)
|
|
{
|
|
if (GetEPPMode () == 32)
|
|
{
|
|
EPPWrite32Buffer (0x400, dest);
|
|
DBG (16,
|
|
"Loop %d: EPPWrite32Buffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
EPPWriteBuffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPWriteBuffer(0x400,dest) passed... (%s:%d)\n",
|
|
i, __FILE__, __LINE__);
|
|
}
|
|
}
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
REGISTERWRITE (0x0A, 0x11);
|
|
for (i = 0; i < 150; i++)
|
|
{
|
|
if (GetEPPMode () == 32)
|
|
{
|
|
EPPRead32Buffer (0x400, dest);
|
|
}
|
|
else
|
|
{
|
|
EPPReadBuffer (0x400, dest);
|
|
}
|
|
for (j = 0; j < 256; j++)
|
|
{
|
|
if (dest[j * 2] != j)
|
|
{
|
|
DBG (0,
|
|
"Altered buffer value at %03X, expected %02X, found %02X\n",
|
|
j * 2, j, dest[j * 2]);
|
|
return (0);
|
|
}
|
|
if (dest[j * 2 + 1] != 0xFF - j)
|
|
{
|
|
DBG
|
|
(0,
|
|
"Altered buffer value at %03X, expected %02X, found %02X\n",
|
|
j * 2 + 1, 0xFF - j, dest[j * 2 + 1]);
|
|
return (0);
|
|
}
|
|
if (dest[512 + j * 2] != j)
|
|
{
|
|
DBG (0,
|
|
"Altered buffer value at %03X, expected %02X, found %02X\n",
|
|
512 + j * 2, j, dest[512 + j * 2]);
|
|
return (0);
|
|
}
|
|
if (dest[512 + j * 2 + 1] != 0xFF - j)
|
|
{
|
|
DBG
|
|
(0,
|
|
"Altered buffer value at %03X, expected 0x%02X, found 0x%02X\n",
|
|
512 + j * 2 + 1, 0xFF - j, dest[512 + j * 2 + 1]);
|
|
return (0);
|
|
}
|
|
}
|
|
if (GetEPPMode () == 32)
|
|
DBG (16, "Loop %d: EPPRead32Buffer(0x400,dest) passed... (%s:%d)\n",
|
|
i, __FILE__, __LINE__);
|
|
else
|
|
DBG (16, "Loop %d: EPPReadBuffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
if (Fonc001 () != 1)
|
|
{
|
|
DBG (0, "Fonc001() failed ! (%s:%d) \n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "Fonc001() passed ... (%s:%d) \n", __FILE__, __LINE__);
|
|
|
|
/* sync */
|
|
if (SendWord (zero) == 0)
|
|
{
|
|
DBG (0, "SendWord(zero) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendWord(zero) passed (%s:%d)\n", __FILE__, __LINE__);
|
|
Epilogue ();
|
|
|
|
/* OK ! */
|
|
free (dest);
|
|
DBG (1, "InitTransport1220P done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
/*
|
|
1: OK
|
|
2: failed, try again
|
|
0: init failed
|
|
|
|
initialize the transport layer
|
|
|
|
*/
|
|
|
|
int
|
|
sanei_umax_pp_InitTransport (int recover)
|
|
{
|
|
switch (sanei_umax_pp_getastra ())
|
|
{
|
|
case 610:
|
|
return (InitTransport610P (recover));
|
|
case 1220:
|
|
case 1600:
|
|
case 2000:
|
|
default:
|
|
return (InitTransport1220P (recover));
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* returns 1 if testing OK
|
|
* 0 if it fails
|
|
* send value if value != 0
|
|
*/
|
|
static int
|
|
Test610P (int value)
|
|
{
|
|
int data, control, val;
|
|
|
|
/* save state */
|
|
data = Inb (DATA);
|
|
control = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, control);
|
|
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0x22);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0xAA);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x55);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0x00);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
Outb (DATA, 0xFF);
|
|
usleep (10000);
|
|
if (value)
|
|
{
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
Outb (DATA, value);
|
|
usleep (10000);
|
|
}
|
|
val = Inb (STATUS);
|
|
usleep (10000);
|
|
Outb (DATA, data);
|
|
Outb (CONTROL, control);
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* returns 1 if testing OK
|
|
* 0 if it fails
|
|
*/
|
|
static int
|
|
In256 (void)
|
|
{
|
|
int val, i, tmp;
|
|
|
|
Outb (CONTROL, 0x04);
|
|
usleep (10000);
|
|
Outb (CONTROL, 0x0C);
|
|
usleep (10000);
|
|
val = Inb (STATUS);
|
|
Outb (CONTROL, 0x0E);
|
|
usleep (10000);
|
|
Outb (CONTROL, 0x0E);
|
|
usleep (10000);
|
|
Outb (CONTROL, 0x0E);
|
|
usleep (10000);
|
|
tmp = val;
|
|
i = 0;
|
|
while ((tmp == val) && (i < 256))
|
|
{
|
|
tmp = Inb (STATUS);
|
|
usleep (10000);
|
|
i++;
|
|
}
|
|
if (tmp != val)
|
|
{
|
|
DBG (1, "Error, expected status 0x%02X, got 0x%02X (%s:%d)\n", val, tmp,
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
Outb (CONTROL, 0x04);
|
|
usleep (10000);
|
|
Outb (CONTROL, 0x04);
|
|
usleep (10000);
|
|
return (1);
|
|
}
|
|
|
|
/* 1: OK
|
|
0: probe failed */
|
|
|
|
static int
|
|
Probe610P (int recover)
|
|
{
|
|
int tmp, i;
|
|
/*int zero[5] = { 0, 0, 0, 0, -1 }; to be used */
|
|
|
|
/* recover is not used yet, but will */
|
|
recover = 0; /* quit compiler quiet .. */
|
|
|
|
/* make sure we won't try 1220/200P later */
|
|
if (!sanei_umax_pp_getastra ())
|
|
sanei_umax_pp_setastra (610);
|
|
if (!Test610P (0x87))
|
|
{
|
|
DBG (1, "Test610P(0x87) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
DBG (16, "Test610P(0x87) passed...\n");
|
|
}
|
|
if (!In256 ())
|
|
{
|
|
DBG (1, "In256() failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
DBG (16, "In256() passed...\n");
|
|
}
|
|
|
|
|
|
/* test PS2 mode */
|
|
|
|
tmp = SlowNibbleRegisterRead (0x0B);
|
|
if (tmp != 0x88)
|
|
{ /* tmp = 0x88 for 610P */
|
|
DBG (1, "Found 0x%X expected 0x88 (%s:%d)\n", tmp, __FILE__, __LINE__);
|
|
}
|
|
|
|
/* clear register 3 */
|
|
ClearRegister (3);
|
|
DBG (16, "ClearRegister(3) passed...\n");
|
|
|
|
/* wait ? */
|
|
i = 65535;
|
|
while (i > 0)
|
|
{
|
|
tmp = Inb (DATA);
|
|
tmp = Inb (DATA);
|
|
i--;
|
|
}
|
|
DBG (16, "FFFF in loop passed...\n");
|
|
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
fflush (stdout);
|
|
|
|
/* no EPP32 for 610P */
|
|
SetEPPMode (8);
|
|
|
|
/* XXX STEF XXX : insert init trans + re homing */
|
|
|
|
/* successfull end ... */
|
|
DBG (1, "UMAX Astra 610P detected\n");
|
|
DBG (1, "Probe610P done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* 1: OK
|
|
0: probe failed */
|
|
|
|
int
|
|
sanei_umax_pp_ProbeScanner (int recover)
|
|
{
|
|
int tmp, i, j;
|
|
int reg;
|
|
unsigned char *dest = NULL;
|
|
int initbuf[2049];
|
|
int voidbuf[2049];
|
|
int val;
|
|
int zero[5] = { 0, 0, 0, 0, -1 };
|
|
int model;
|
|
|
|
/* save and set CONTROL */
|
|
tmp = (Inb (CONTROL)) & 0x1F;
|
|
tmp = (Inb (CONTROL)) & 0x1F;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
|
|
tmp = Inb (DATA);
|
|
tmp = Inb (CONTROL) & 0x3F;
|
|
Outb (CONTROL, tmp);
|
|
|
|
tmp = Inb (CONTROL) & 0x3F;
|
|
tmp = Inb (DATA);
|
|
tmp = Inb (CONTROL) & 0x3F;
|
|
tmp = Inb (DATA);
|
|
|
|
/* any scanner ? */
|
|
/* fast detect */
|
|
tmp = RingScanner (2, 0);
|
|
if (!tmp)
|
|
{
|
|
DBG (1, "No scanner detected by 'RingScanner(2,0)'...\n");
|
|
tmp = RingScanner (5, 0);
|
|
if (!tmp)
|
|
{
|
|
DBG (1, "No scanner detected by 'RingScanner(5,0)'...\n");
|
|
tmp = RingScanner (5, 10000);
|
|
if (!tmp)
|
|
{
|
|
DBG (1, "No scanner detected by 'RingScanner(5,10000)'...\n");
|
|
tmp = RingScanner (5, 10000);
|
|
if (!tmp)
|
|
{
|
|
DBG (1,
|
|
"No scanner detected by 'RingScanner(5,10000)'...\n");
|
|
tmp = Test610P (0x87);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (!tmp)
|
|
{
|
|
DBG (1, "No 1220P/2000P scanner detected by 'RingScanner()'...\n");
|
|
DBG (1, "Trying 610P...\n");
|
|
return (Probe610P (recover));
|
|
}
|
|
DBG (16, "RingScanner passed...\n");
|
|
|
|
gControl = Inb (CONTROL) & 0x3F;
|
|
g67D = 1;
|
|
if (SendCommand (0x30) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0x30) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0x30) passed ... (%s:%d)\n", __FILE__, __LINE__);
|
|
g67E = 4; /* bytes to read */
|
|
if (SendCommand (0x00) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0x00) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0x00) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
g67E = 0; /* bytes to read */
|
|
if (TestVersion (0) == 0)
|
|
{
|
|
DBG (0, "TestVersion(0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "TestVersion(0) passed...\n");
|
|
/* must fail for 1220P and 2000P */
|
|
if (TestVersion (1) == 0) /* software doesn't do it for model 0x07 */
|
|
{ /* but it works .. */
|
|
DBG (16, "TestVersion(1) failed (expected) ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "Unexpected success on TestVersion(1) ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
if (TestVersion (0) == 0)
|
|
{
|
|
DBG (0, "TestVersion(0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "TestVersion(0) passed...\n");
|
|
/* must fail */
|
|
if (TestVersion (1) == 0)
|
|
{
|
|
DBG (16, "TestVersion(1) failed (expected) ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "Unexpected success on TestVersion(1) ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
Outb (DATA, 0x04);
|
|
Outb (CONTROL, 0x0C);
|
|
Outb (DATA, 0x04);
|
|
Outb (CONTROL, 0x0C);
|
|
|
|
gControl = Inb (CONTROL) & 0x3F;
|
|
Outb (CONTROL, gControl & 0xEF);
|
|
|
|
|
|
|
|
if (SendCommand (0x40) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0x40) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0x40) passed...\n");
|
|
if (SendCommand (0xE0) == 0)
|
|
{
|
|
DBG (0, "SendCommand(0xE0) (%s:%d) failed ...\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "SendCommand(0xE0) passed...\n");
|
|
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed...\n");
|
|
|
|
SPPResetLPT ();
|
|
DBG (16, "SPPResetLPT() passed...\n");
|
|
|
|
Outb (CONTROL, 4);
|
|
Outb (CONTROL, 4);
|
|
|
|
|
|
/* test PS2 mode */
|
|
|
|
tmp = SlowNibbleRegisterRead (0x0B);
|
|
if (tmp == 0xC7)
|
|
{
|
|
/* epat C7 detected */
|
|
DBG (16, "SlowNibbleRegisterRead(0x0B)=0x%X passed...\n", tmp);
|
|
|
|
WriteSlow (8, 0);
|
|
DBG (16, "WriteSlow(8,0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
tmp = SlowNibbleRegisterRead (0x0A);
|
|
if (tmp != 0x00)
|
|
{
|
|
if (tmp == 0x1C)
|
|
{
|
|
DBG (16, "Previous probe detected ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "Found 0x%X expected 0x00 (%s:%d)\n", tmp, __FILE__,
|
|
__LINE__);
|
|
}
|
|
}
|
|
DBG (16, "SlowNibbleRegisterRead(0x0A)=0x%X passed ...(%s:%d)\n", tmp,
|
|
__FILE__, __LINE__);
|
|
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "Found 0x%X expected 0xC7 or 0x88 (%s:%d)\n", tmp, __FILE__,
|
|
__LINE__);
|
|
return 0;
|
|
}
|
|
|
|
/* clear register 3 */
|
|
ClearRegister (3);
|
|
DBG (16, "ClearRegister(3) passed...\n");
|
|
|
|
/* wait ? */
|
|
i = 65535;
|
|
while (i > 0)
|
|
{
|
|
tmp = Inb (DATA);
|
|
tmp = Inb (DATA);
|
|
i--;
|
|
}
|
|
DBG (16, "FFFF in loop passed...\n");
|
|
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
fflush (stdout);
|
|
|
|
/* 1220/2000P branch */
|
|
WRITESLOW (0x0E, 1);
|
|
|
|
/* register 0x0F used only once: model number ? Or ASIC revision ? */
|
|
/* comm mode ? */
|
|
model = SlowNibbleRegisterRead (0x0F);
|
|
DBG (1, "UMAX Astra 1220/1600/2000 P ASIC detected (mode=%d)\n", model);
|
|
SetModel (model);
|
|
DBG (16, "SlowNibbleRegisterRead(0x0F) passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
/* scanner powered off */
|
|
if (model == 0x1B)
|
|
{
|
|
DBG (0, "Register 0x0F=0x1B, scanner powered off ! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return 0;
|
|
}
|
|
|
|
|
|
if ((model != 0x1F) && (model != 0x07))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Unexpected value for for register 0x0F, expected 0x07 or 0x1F, got 0x%02X ! (%s:%d)\n",
|
|
model, __FILE__, __LINE__);
|
|
DBG (0, "There is a new scanner revision in town, or a bug ....\n");
|
|
}
|
|
|
|
WRITESLOW (0x08, 0x02);
|
|
WRITESLOW (0x0E, 0x0F);
|
|
WRITESLOW (0x0F, 0x0C);
|
|
WRITESLOW (0x0C, 0x04);
|
|
tmp = SlowNibbleRegisterRead (0x0D);
|
|
if ((tmp != 0x00) && (tmp != 0x40))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Unexpected value for for register 0x0D, expected 0x00 or 0x40, got 0x%02X ! (%s:%d)\n",
|
|
tmp, __FILE__, __LINE__);
|
|
}
|
|
WRITESLOW (0x0D, 0x1B);
|
|
switch (model)
|
|
{
|
|
case 0x1F:
|
|
WRITESLOW (0x12, 0x14);
|
|
SLOWNIBBLEREGISTEREAD (0x12, 0x10);
|
|
break;
|
|
case 0x07:
|
|
WRITESLOW (0x12, 0x00);
|
|
SLOWNIBBLEREGISTEREAD (0x12, 0x00);
|
|
/* we may get 0x20, in this case some color aberration may occur */
|
|
/* must depend on the parport */
|
|
/* model 0x07 + 0x00=>0x20=2000P */
|
|
break;
|
|
default:
|
|
WRITESLOW (0x12, 0x00);
|
|
SLOWNIBBLEREGISTEREAD (0x12, 0x20);
|
|
break;
|
|
}
|
|
SLOWNIBBLEREGISTEREAD (0x0D, 0x18);
|
|
SLOWNIBBLEREGISTEREAD (0x0C, 0x04);
|
|
SLOWNIBBLEREGISTEREAD (0x0A, 0x00);
|
|
WRITESLOW (0x0E, 0x0A);
|
|
WRITESLOW (0x0F, 0x00);
|
|
WRITESLOW (0x0E, 0x0D);
|
|
WRITESLOW (0x0F, 0x00);
|
|
|
|
/* write/read full buffer */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
WRITESLOW (0x0A, i);
|
|
SLOWNIBBLEREGISTEREAD (0x0A, i);
|
|
WRITESLOW (0x0A, 0xFF - i);
|
|
SLOWNIBBLEREGISTEREAD (0x0A, 0xFF - i);
|
|
}
|
|
|
|
/* end test for nibble byte/byte mode */
|
|
|
|
/* now we try nibble buffered mode */
|
|
WRITESLOW (0x13, 0x01);
|
|
WRITESLOW (0x13, 0x00); /*reset something */
|
|
WRITESLOW (0x0A, 0x11);
|
|
|
|
/* read buffer */
|
|
dest = (unsigned char *) (malloc (65536));
|
|
for (i = 0; i < 10; i++) /* 10 ~ 11 ? */
|
|
{
|
|
NibbleReadBuffer (0x400, dest);
|
|
DBG (16, "Loop %d: NibbleReadBuffer passed ... (%s:%d)\n", i, __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
/* write buffer */
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
WriteBuffer (0x400, dest);
|
|
DBG (16, "Loop %d: WriteBuffer passed ... (%s:%d)\n", i, __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
SLOWNIBBLEREGISTEREAD (0x0C, 0x04);
|
|
WRITESLOW (0x13, 0x01);
|
|
WRITESLOW (0x13, 0x00);
|
|
WRITESLOW (0x0A, 0x18);
|
|
Outb (CONTROL, 4);
|
|
SLOWNIBBLEREGISTEREAD (0x0A, 0x18);
|
|
WRITESLOW (0x08, 0x40);
|
|
WRITESLOW (0x08, 0x60);
|
|
WRITESLOW (0x08, 0x22);
|
|
|
|
|
|
/* test EPP MODE */
|
|
SetEPPMode (8);
|
|
ClearRegister (0);
|
|
DBG (16, "ClearRegister(0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
WRITESLOW (0x08, 0x22);
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
gEPAT = 0xC7;
|
|
Init002 (0);
|
|
DBG (16, "Init002(0) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
REGISTERWRITE (0x0A, 0);
|
|
|
|
/* catch any failure to read back data in EPP mode */
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0)
|
|
{
|
|
DBG (0, "RegisterRead, found 0x%X expected 0x00 (%s:%d)\n", reg,
|
|
__FILE__, __LINE__);
|
|
if (reg == 0xFF)
|
|
{
|
|
/* EPP mode not set: try ECP if is available */
|
|
/* reset EPAT */
|
|
reg = Inb (CONTROL) & 0x04;
|
|
Outb (CONTROL, reg);
|
|
Outb (ECPCONTROL, 0x34);
|
|
Outb (CONTROL, reg);
|
|
reg = reg | 0x08;
|
|
Outb (CONTROL, reg);
|
|
Outb (CONTROL, reg);
|
|
Outb (CONTROL, reg);
|
|
Outb (CONTROL, reg);
|
|
for (i = 0; i < 256; i++)
|
|
reg = Inb (STATUS);
|
|
|
|
DBG (0,
|
|
"*** It appears that EPP data transfer doesn't work ***\n");
|
|
DBG (0,
|
|
"*** Please read SETTING EPP section in sane-umax_pp.5 ***\n");
|
|
}
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
gMode = UMAX_PP_PARPORT_EPP;
|
|
DBG (16, "RegisterRead(0x0A)=0x00 passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
RegisterWrite (0x0A, 0xFF);
|
|
DBG (16, "RegisterWrite(0x%X,0x%X) passed... (%s:%d)\n", 0x0A, 0xFF,
|
|
__FILE__, __LINE__);
|
|
REGISTERREAD (0x0A, 0xFF);
|
|
for (i = 1; i < 256; i++)
|
|
{
|
|
REGISTERWRITE (0x0A, i);
|
|
REGISTERREAD (0x0A, i);
|
|
REGISTERWRITE (0x0A, 0xFF - i);
|
|
REGISTERREAD (0x0A, 0xFF - i);
|
|
}
|
|
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x11);
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
EPPReadBuffer (0x400, dest);
|
|
for (j = 0; j < 512; j++)
|
|
{
|
|
if (dest[2 * j] != (j % 256))
|
|
{
|
|
DBG (0, "Loop %d, char %d EPPReadBuffer failed! (%s:%d)\n", i,
|
|
j * 2, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
if (dest[2 * j + 1] != (0xFF - (j % 256)))
|
|
{
|
|
DBG (0, "Loop %d, char %d EPPReadBuffer failed! (%s:%d)\n", i,
|
|
j * 2 + 1, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
DBG (16, "Loop %d: EPPReadBuffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
EPPWriteBuffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPWriteBuffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
|
|
REGISTERREAD (0x0C, 4);
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
|
|
Outb (DATA, 0x0);
|
|
ClearRegister (0);
|
|
Init001 ();
|
|
|
|
if (CheckEPAT () != 0)
|
|
return (0);
|
|
DBG (16, "CheckEPAT() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
tmp = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, tmp);
|
|
Outb (CONTROL, tmp);
|
|
|
|
WRITESLOW (0x68, 0x21);
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
WRITESLOW (0x68, 0x21);
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
SPPResetLPT ();
|
|
|
|
|
|
if (Init005 (0x80))
|
|
{
|
|
DBG (0, "Init005(0x80) failed... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "Init005(0x80) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
if (Init005 (0xEC))
|
|
{
|
|
DBG (0, "Init005(0xEC) failed... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "Init005(0xEC) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
/* write/read buffer loop */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
REGISTERWRITE (0x0A, i);
|
|
REGISTERREAD (0x0A, i);
|
|
REGISTERWRITE (0x0A, 0xFF - i);
|
|
REGISTERREAD (0x0A, 0xFF - i);
|
|
}
|
|
DBG (16, "EPP write/read buffer loop passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x11);
|
|
|
|
/* test EPP32 mode */
|
|
/* we set 32 bits I/O mode first, then step back to */
|
|
/* 8bits if tests fail */
|
|
SetEPPMode (32);
|
|
for (i = 0; (i < 10) && (GetEPPMode () == 32); i++)
|
|
{
|
|
EPPRead32Buffer (0x400, dest);
|
|
/* if 32 bit I/O work, we should have a buffer */
|
|
/* filled by 00 FF 01 FE 02 FD 03 FC ..... */
|
|
for (j = 0; j < 0x200; j++)
|
|
{
|
|
if ((dest[j * 2] != j % 256)
|
|
|| (dest[j * 2 + 1] != 0xFF - (j % 256)))
|
|
{
|
|
DBG (1, "Setting EPP I/O to 8 bits ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
SetEPPMode (8);
|
|
/* leave out current loop since an error was detected */
|
|
break;
|
|
}
|
|
}
|
|
DBG (16, "Loop %d: EPPRead32Buffer(0x400) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
DBG (1, "%d bits EPP data transfer\n", GetEPPMode ());
|
|
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
EPPWrite32Buffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPWrite32Buffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
|
|
|
|
REGISTERREAD (0x0C, 0x04);
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
WRITESLOW (0x68, 0x21);
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
SPPResetLPT ();
|
|
|
|
if (Init005 (0x80))
|
|
{
|
|
DBG (0, "Init005(0x80) failed... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "Init005(0x80) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
if (Init005 (0xEC))
|
|
{
|
|
DBG (0, "Init005(0xEC) failed... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
DBG (16, "Init005(0xEC) passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
/* write/read buffer loop */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
REGISTERWRITE (0x0A, i);
|
|
REGISTERREAD (0x0A, i);
|
|
REGISTERWRITE (0x0A, 0xFF - i);
|
|
REGISTERREAD (0x0A, 0xFF - i);
|
|
}
|
|
DBG (16, "EPP write/read buffer loop passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x11);
|
|
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
EPPRead32Buffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPRead32Buffer(0x400) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
EPPWrite32Buffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPWrite32Buffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERREAD (0x0C, 0x04);
|
|
REGISTERWRITE (0x13, 0x01);
|
|
REGISTERWRITE (0x13, 0x00);
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
WRITESLOW (0x68, 0x21);
|
|
g6FE = 1;
|
|
Init001 ();
|
|
DBG (16, "Init001() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
|
|
REGISTERWRITE (0x0A, 0x18);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
|
|
|
|
reg = RegisterRead (0x0D);
|
|
reg = RegisterRead (0x0D);
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xB7) | 0x03;
|
|
RegisterWrite (0x0D, reg);
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
|
|
reg = RegisterRead (0x12); /* 0x10 for model 0x0F, 0x20 for model 0x07 */
|
|
reg = reg & 0xEF;
|
|
RegisterWrite (0x12, reg);
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x1C)
|
|
{
|
|
DBG (0, "Warning! expected reg0A=0x1C, found 0x%02X! (%s:%d) \n", reg,
|
|
__FILE__, __LINE__);
|
|
}
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
Init021 ();
|
|
DBG (16, "Init021() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
/* some sort of countdown, some warming-up ? */
|
|
/* maybe some pauses are needed */
|
|
/* if (model == 0x07) */
|
|
{
|
|
REGISTERWRITE (0x0A, 0x00);
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8);
|
|
RegisterWrite (0x0D, reg);
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
Init022 ();
|
|
DBG (16, "Init022() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
DBG (0, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n",
|
|
gEPAT, reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
DBG (0, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n",
|
|
reg, __FILE__, __LINE__);
|
|
}
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
reg = RegisterRead (0x13);
|
|
if (reg != 0x00)
|
|
{
|
|
DBG (0, "Warning! expected reg13=0x00, found 0x%02X! (%s:%d) \n",
|
|
reg, __FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x13, 0x81);
|
|
usleep (10000);
|
|
REGISTERWRITE (0x13, 0x80);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0xFF);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x03);
|
|
REGISTERWRITE (0x10, 0x66);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0xFF);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x01);
|
|
REGISTERWRITE (0x10, 0x55);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0xFF);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x10, 0x44);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0x7F);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x10, 0x33);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0x3F);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x10, 0x22);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x10, 0x11);
|
|
usleep (10000);
|
|
|
|
REGISTERWRITE (0x13, 0x81);
|
|
usleep (10000);
|
|
REGISTERWRITE (0x13, 0x80);
|
|
|
|
REGISTERWRITE (0x0E, 0x04);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x0E, 0x05);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
usleep (10000);
|
|
|
|
reg = RegisterRead (0x10);
|
|
DBG (1, "Count-down value is 0x%02X (%s:%d)\n", reg, __FILE__, __LINE__);
|
|
/* 2 reports of CF, was FF first (typo ?) */
|
|
/* CF seems a valid value */
|
|
/* in case of CF, we may have timeout ... */
|
|
/*if (reg != 0x00)
|
|
{
|
|
DBG (0, "Warning! expected reg10=0x00, found 0x%02X! (%s:%d) \n",
|
|
reg, __FILE__, __LINE__);
|
|
} */
|
|
REGISTERWRITE (0x13, 0x00);
|
|
}
|
|
|
|
REGISTERWRITE (0x0A, 0x00);
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8);
|
|
RegisterWrite (0x0D, reg);
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
Init022 ();
|
|
DBG (16, "Init022() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
DBG (0, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n", gEPAT,
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
DBG (0, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n", reg,
|
|
__FILE__, __LINE__);
|
|
}
|
|
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
|
|
/* real start of high level protocol ? */
|
|
if (Fonc001 () != 1)
|
|
{
|
|
DBG (0, "Fonc001() failed ! (%s:%d) \n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "Fct001() passed (%s:%d) \n", __FILE__, __LINE__);
|
|
reg = RegisterRead (0x19) & 0xC8;
|
|
/* if reg=E8 or D8 , we have a 'messed' scanner */
|
|
|
|
/* 4 tranform buffers + 'void' are sent: 1 B&W, and 3 RGB ? */
|
|
memset (initbuf, 0x00, 2048 * sizeof (int));
|
|
memset (voidbuf, 0x00, 2048 * sizeof (int));
|
|
|
|
initbuf[512] = 0xFF;
|
|
initbuf[513] = 0xAA;
|
|
initbuf[514] = 0x55;
|
|
|
|
for (j = 0; j < 4; j++)
|
|
{
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
voidbuf[512 * j + 2 * i] = i;
|
|
voidbuf[512 * j + 2 * i] = 0xFF - i;
|
|
}
|
|
}
|
|
|
|
/* one pass (B&W ?) */
|
|
if (CmdSetDataBuffer (initbuf) != 1)
|
|
{
|
|
DBG (0, "CmdSetDataBuffer(initbuf) failed ! (%s:%d) \n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetDataBuffer(initbuf) passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
if (CmdSetDataBuffer (voidbuf) != 1)
|
|
{
|
|
DBG (0, "CmdSetDataBuffer(voidbuf) failed ! (%s:%d) \n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetDataBuffer(voidbuf) passed... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
|
|
/* everything above the FF 55 AA tag is 'void' */
|
|
/* it seems that the buffer is reused and only the beginning is initalized */
|
|
for (i = 515; i < 2048; i++)
|
|
initbuf[i] = voidbuf[i];
|
|
|
|
/* three pass (RGB ?) */
|
|
for (i = 0; i < 3; i++)
|
|
{
|
|
if (CmdSetDataBuffer (initbuf) != 1)
|
|
{
|
|
DBG (0, "CmdSetDataBuffer(initbuf) failed ! (%s:%d) \n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "Loop %d: CmdSetDataBuffer(initbuf) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
if (CmdSetDataBuffer (voidbuf) != 1)
|
|
{
|
|
DBG (0, "Loop %d: CmdSetDataBuffer(voidbuf) failed ! (%s:%d) \n", i,
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
|
|
/* memory size testing ? */
|
|
/* load 150 Ko in scanner */
|
|
REGISTERWRITE (0x1A, 0x00);
|
|
REGISTERWRITE (0x1A, 0x0C);
|
|
REGISTERWRITE (0x1A, 0x00);
|
|
REGISTERWRITE (0x1A, 0x0C);
|
|
|
|
REGISTERWRITE (0x0A, 0x11); /* start */
|
|
for (i = 0; i < 150; i++)
|
|
{
|
|
EPPWrite32Buffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPWrite32Buffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x18); /* end */
|
|
|
|
/* read them back */
|
|
REGISTERWRITE (0x0A, 0x11); /*start transfert */
|
|
for (i = 0; i < 150; i++)
|
|
{
|
|
EPPRead32Buffer (0x400, dest);
|
|
DBG (16, "Loop %d: EPPRead32Buffer(0x400,dest) passed... (%s:%d)\n", i,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x18); /*end transfer */
|
|
|
|
|
|
|
|
/* almost CmdSync(0x00) which halts any pending operation */
|
|
if (Fonc001 () != 1)
|
|
{
|
|
DBG (0, "Fonc001() failed ! (%s:%d) \n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "Fct001() passed (%s:%d) \n", __FILE__, __LINE__);
|
|
if (SendWord (zero) == 0)
|
|
{
|
|
DBG (0, "SendWord(zero) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
Epilogue ();
|
|
DBG (16, "SendWord(zero) passed (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
/* end transport init */
|
|
/* now high level (connected) protocol begins */
|
|
val = sanei_umax_pp_InitScanner (recover);
|
|
if (val == 0)
|
|
{
|
|
DBG (0, "InitScanner() failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
|
|
/* if no homing .... */
|
|
if (val == 1)
|
|
{
|
|
CMDSYNC (0);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0);
|
|
}
|
|
|
|
/* set port to its initial state */
|
|
Outb (DATA, 0x04);
|
|
Outb (CONTROL, 0x0C);
|
|
|
|
free (dest);
|
|
DBG (1, "Probe done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
static int
|
|
deconnect_epat (void)
|
|
{
|
|
int tmp;
|
|
|
|
REGISTERWRITE (0x0A, 0x00);
|
|
REGISTERREAD (0x0D, 0x40);
|
|
REGISTERWRITE (0x0D, 0x00);
|
|
if (GetModel () != 0x07)
|
|
SendCommand (40);
|
|
SendCommand (30);
|
|
Outb (DATA, gData);
|
|
Outb (CONTROL, gControl);
|
|
return (1);
|
|
}
|
|
|
|
|
|
static int
|
|
connect_epat (void)
|
|
{
|
|
int reg;
|
|
|
|
|
|
gData = Inb (DATA);
|
|
gControl = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, gControl);
|
|
gControl = Inb (CONTROL) & 0x1F;
|
|
Outb (CONTROL, gControl);
|
|
if (SendCommand (0xE0) != 1)
|
|
{
|
|
DBG (0, "connect_epat: SendCommand(0xE0) failed! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
/* we try to clean all */
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
Outb (DATA, 0x00);
|
|
Outb (CONTROL, 0x01);
|
|
Outb (CONTROL, 0x04);
|
|
ClearRegister (0);
|
|
Init001 ();
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
/* ASIC version is not */
|
|
/* the one expected (epat c7) */
|
|
DBG (0, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n", gEPAT,
|
|
reg, __FILE__, __LINE__);
|
|
/* we try to clean all */
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg | 0x43) & 0xEB;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
/* a previous unfinished command */
|
|
/* has left an uncleared value */
|
|
DBG (0, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n", reg,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
connect_610P (void)
|
|
{
|
|
int tmp;
|
|
|
|
gData = Inb (DATA);
|
|
Outb (DATA, 0xAA);
|
|
Outb (CONTROL, 0x0E);
|
|
Inb (CONTROL);
|
|
Inb (CONTROL);
|
|
Outb (DATA, 0x00);
|
|
Outb (CONTROL, 0x0C);
|
|
Inb (CONTROL);
|
|
Inb (CONTROL);
|
|
Outb (DATA, 0x55);
|
|
Outb (CONTROL, 0x0E);
|
|
Inb (CONTROL);
|
|
Inb (CONTROL);
|
|
Outb (DATA, 0xFF);
|
|
Outb (CONTROL, 0x0C);
|
|
Inb (CONTROL);
|
|
Inb (CONTROL);
|
|
Outb (CONTROL, 0x04);
|
|
Inb (CONTROL);
|
|
Inb (CONTROL);
|
|
Outb (DATA, 0x40);
|
|
Outb (CONTROL, 0x06);
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0x38)
|
|
{
|
|
DBG (0, "Error! expected STATUS=0x38, found 0x%02X! (%s:%d) \n", tmp,
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
Outb (CONTROL, 0x07);
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0x38)
|
|
{
|
|
DBG (0, "Error! expected STATUS=0x38, found 0x%02X! (%s:%d) \n", tmp,
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
Outb (CONTROL, 0x04);
|
|
tmp = Inb (STATUS);
|
|
if (tmp != 0xF8)
|
|
{
|
|
DBG (0, "Error! expected STATUS=0xF8, found 0x%02X! (%s:%d) \n", tmp,
|
|
__FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
|
|
static int
|
|
deconnect_610P (void)
|
|
{
|
|
int tmp, i;
|
|
|
|
Outb (CONTROL, 0x04);
|
|
i = 0;
|
|
do
|
|
{
|
|
i++;
|
|
tmp = Inb (CONTROL);
|
|
if (tmp != 0x04)
|
|
{
|
|
DBG (0, "Error! expected CONTROL=0x04, found 0x%02X! (%s:%d) \n",
|
|
tmp, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
while ((i < 41) && (tmp == 0x04));
|
|
|
|
Outb (DATA, gData);
|
|
return (1);
|
|
}
|
|
|
|
static int
|
|
Prologue (void)
|
|
{
|
|
switch (sanei_umax_pp_getastra ())
|
|
{
|
|
case 610:
|
|
return (connect_610P ());
|
|
case 1220:
|
|
case 1600:
|
|
case 2000:
|
|
default:
|
|
return (connect_epat ());
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
Epilogue (void)
|
|
{
|
|
switch (sanei_umax_pp_getastra ())
|
|
{
|
|
case 610:
|
|
return (deconnect_610P ());
|
|
case 1220:
|
|
case 1600:
|
|
case 2000:
|
|
default:
|
|
return (deconnect_epat ());
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
CmdSet (int cmd, int len, int *val)
|
|
{
|
|
int word[5];
|
|
int i;
|
|
|
|
/* cmd 08 as 2 length depending upon model */
|
|
if ((cmd == 8) && (GetModel () == 0x07))
|
|
{
|
|
len = 35;
|
|
}
|
|
|
|
/* compute word */
|
|
word[0] = len / 65536;
|
|
word[1] = len / 256 % 256;
|
|
word[2] = len % 256;
|
|
word[3] = (cmd & 0x3F) | 0x80;
|
|
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdSet: Prologue failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* send data */
|
|
if (SendLength (word, 4) == 0)
|
|
{
|
|
DBG (0, "SendLength(word,4) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendLength(word,4) passed ...");
|
|
|
|
/* head end */
|
|
Epilogue ();
|
|
|
|
if (len > 0)
|
|
{
|
|
/* send body */
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdSet: Prologue failed ! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
if (DBG_LEVEL >= 8)
|
|
{
|
|
char *str = NULL;
|
|
|
|
str = malloc (3 * len + 1);
|
|
if (str != NULL)
|
|
{
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
sprintf (str + 3 * i, "%02X ", val[i]);
|
|
}
|
|
str[3 * i] = 0x00;
|
|
DBG (8, "String sent for %02X: %s\n", cmd, str);
|
|
free (str);
|
|
}
|
|
else
|
|
{
|
|
TRACE (8, "not enough memory for debugging ...");
|
|
}
|
|
}
|
|
|
|
/* send data */
|
|
if (SendData (val, len) == 0)
|
|
{
|
|
DBG (0, "SendData(word,%d) failed (%s:%d)\n", len, __FILE__,
|
|
__LINE__);
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendData(val,len) passed ...");
|
|
/* body end */
|
|
Epilogue ();
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
static int
|
|
CmdGet (int cmd, int len, int *val)
|
|
{
|
|
int word[5];
|
|
int i;
|
|
|
|
/* cmd 08 as 2 length depending upon model */
|
|
if ((cmd == 8) && (GetModel () == 0x07))
|
|
{
|
|
len = 35;
|
|
}
|
|
|
|
/* compute word */
|
|
word[0] = len / 65536;
|
|
word[1] = len / 256 % 256;
|
|
word[2] = len % 256;
|
|
word[3] = (cmd & 0x3F) | 0x80 | 0x40; /* 0x40 means 'read' */
|
|
word[4] = -1;
|
|
|
|
/* send header */
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdGet: Prologue failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* send data */
|
|
if (SendLength (word, 4) == 0)
|
|
{
|
|
DBG (0, "SendLength(word,4) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendLength(word,4) passed ...");
|
|
|
|
/* head end */
|
|
Epilogue ();
|
|
|
|
|
|
/* send header */
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdGet: Prologue failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* get actual data */
|
|
if (ReceiveData (val, len) == 0)
|
|
{
|
|
DBG (0, "ReceiveData(val,len) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
if (DBG_LEVEL >= 8)
|
|
{
|
|
char *str = NULL;
|
|
|
|
str = malloc (3 * len + 1);
|
|
if (str != NULL)
|
|
{
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
sprintf (str + 3 * i, "%02X ", val[i]);
|
|
}
|
|
str[3 * i] = 0x00;
|
|
DBG (8, "String received for %02X: %s\n", cmd, str);
|
|
free (str);
|
|
}
|
|
else
|
|
{
|
|
TRACE (8, "not enough memory for debugging ...");
|
|
}
|
|
}
|
|
Epilogue ();
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
CmdSetGet (int cmd, int len, int *val)
|
|
{
|
|
int *tampon;
|
|
int i;
|
|
|
|
/* model revision 0x07 uses 35 bytes buffers */
|
|
/* cmd 08 as 2 length depending upon model */
|
|
if ((cmd == 8) && (GetModel () == 0x07))
|
|
{
|
|
len = 35;
|
|
}
|
|
|
|
/* first we send */
|
|
if (CmdSet (cmd, len, val) == 0)
|
|
{
|
|
DBG (0, "CmdSetGet failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
tampon = (int *) malloc (len * sizeof (int));
|
|
memset (tampon, 0x00, len * sizeof (int));
|
|
if (tampon == NULL)
|
|
{
|
|
DBG (0, "Failed to allocate room for %d int ! (%s:%d)\n", len, __FILE__,
|
|
__LINE__);
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
|
|
/* then we receive */
|
|
if (CmdGet (cmd, len, tampon) == 0)
|
|
{
|
|
DBG (0, "CmdSetGet failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
free (tampon);
|
|
Epilogue ();
|
|
return (0);
|
|
}
|
|
|
|
/* check and copy */
|
|
for (i = 0; (i < len) && (val[i] >= 0); i++)
|
|
{
|
|
if (tampon[i] != val[i])
|
|
{
|
|
DBG
|
|
(0,
|
|
"Warning data read back differs: expected %02X found tampon[%d]=%02X ! (%s:%d)\n",
|
|
val[i], i, tampon[i], __FILE__, __LINE__);
|
|
}
|
|
val[i] = tampon[i];
|
|
}
|
|
|
|
|
|
/* OK */
|
|
free (tampon);
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* 1 OK, 0 failed */
|
|
static int
|
|
CmdGetBuffer (int cmd, int len, unsigned char *buffer)
|
|
{
|
|
int reg, tmp, i;
|
|
int word[5], read;
|
|
int needed;
|
|
|
|
/* compute word */
|
|
word[0] = len / 65536;
|
|
word[1] = len / 256 % 256;
|
|
word[2] = len % 256;
|
|
word[3] = (cmd & 0x3F) | 0x80 | 0x40;
|
|
word[4] = -1;
|
|
|
|
/* send word: len+addr(?) */
|
|
if (FoncSendWord (word) == 0)
|
|
{
|
|
DBG (0, "FoncSendWord(word) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "(%s:%d) passed \n", __FILE__, __LINE__);
|
|
Init022 ();
|
|
DBG (16, "Init022() passed... (%s:%d)\n", __FILE__, __LINE__);
|
|
reg = RegisterRead (0x0B);
|
|
if (reg != gEPAT)
|
|
{
|
|
/* return -1 */
|
|
DBG (0, "Error! expected reg0B=0x%02X, found 0x%02X! (%s:%d) \n", gEPAT,
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
reg = RegisterRead (0x0D);
|
|
reg = (reg & 0xE8) | 0x43;
|
|
RegisterWrite (0x0D, reg);
|
|
REGISTERWRITE (0x0C, 0x04);
|
|
reg = RegisterRead (0x0A);
|
|
if (reg != 0x00)
|
|
{
|
|
DBG (0, "Warning! expected reg0A=0x00, found 0x%02X! (%s:%d) \n", reg,
|
|
__FILE__, __LINE__);
|
|
}
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x08, 0x21);
|
|
REGISTERWRITE (0x0E, 0x0F);
|
|
REGISTERWRITE (0x0F, 0x0C);
|
|
REGISTERWRITE (0x0A, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x10);
|
|
REGISTERWRITE (0x0F, 0x1C);
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
|
|
i = 0;
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* wait if busy */
|
|
while ((reg & 0x08) == 0x08)
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "CmdGetBuffer failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
read = 0;
|
|
reg = RegisterRead (0x0C);
|
|
if (reg != 0x04)
|
|
{
|
|
DBG (0, "CmdGetBuffer failed: unexpected status 0x%02X ...(%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
REGISTERWRITE (0x0C, reg | 0x40);
|
|
|
|
/* actual data */
|
|
read = 0;
|
|
while (read < len)
|
|
{
|
|
needed = len - read;
|
|
if (needed > 32768)
|
|
needed = 32768;
|
|
EPPBlockMode (0x80);
|
|
tmp = PausedReadBuffer (needed, buffer + read);
|
|
if (tmp < needed)
|
|
{
|
|
DBG (64, "CmdGetBuffer only got %d bytes out of %d ...(%s:%d)\n",
|
|
tmp, needed, __FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (64,
|
|
"CmdGetBuffer got all %d bytes out of %d , read=%d...(%s:%d)\n",
|
|
tmp, 32768, read, __FILE__, __LINE__);
|
|
}
|
|
read += tmp;
|
|
DBG (16, "Read %d bytes out of %d (last block is %d bytes) (%s:%d)\n",
|
|
read, len, tmp, __FILE__, __LINE__);
|
|
if (read < len)
|
|
{
|
|
/* wait for scanner to be ready */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
DBG (64, "Status after block read is 0x%02X (%s:%d)\n", reg,
|
|
__FILE__, __LINE__);
|
|
if ((reg & 0x08) == 0x08)
|
|
{
|
|
int pass = 0;
|
|
|
|
do
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
usleep (100);
|
|
pass++;
|
|
}
|
|
while ((pass < 32768) && ((reg & 0x08) == 0x08));
|
|
DBG (64, "Status after waiting is 0x%02X (pass=%d) (%s:%d)\n",
|
|
reg, pass, __FILE__, __LINE__);
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0,
|
|
"Unexpected status 0x%02X, expected 0xC0 or 0xD0 ! (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Going on...\n");
|
|
}
|
|
}
|
|
|
|
/* signal we want next data chunk */
|
|
reg = RegisterRead (0x0C);
|
|
RegisterWrite (0x0C, reg | 0x40);
|
|
}
|
|
}
|
|
|
|
/* OK ! */
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
|
|
/* epilogue */
|
|
Epilogue ();
|
|
return (1);
|
|
}
|
|
|
|
/* 1 OK, 0 failed */
|
|
static int
|
|
CmdGetBuffer32 (int cmd, int len, unsigned char *buffer)
|
|
{
|
|
int reg, tmp, i;
|
|
int word[5], read;
|
|
|
|
/* compute word */
|
|
word[0] = len / 65536;
|
|
word[1] = len / 256 % 256;
|
|
word[2] = len % 256;
|
|
word[3] = (cmd & 0x3F) | 0x80 | 0x40;
|
|
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdSet: Prologue failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* send data */
|
|
if (SendLength (word, 4) == 0)
|
|
{
|
|
DBG (0, "SendLength(word,4) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendLength(word,4) passed ...");
|
|
|
|
/* head end */
|
|
Epilogue ();
|
|
|
|
Prologue ();
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
|
|
i = 0;
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
/* wait if busy */
|
|
while ((reg & 0x08) == 0x08)
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0, "CmdGetBuffer32 failed: unexpected status 0x%02X ...(%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
reg = RegisterRead (0x0C);
|
|
if (reg != 0x04)
|
|
{
|
|
DBG (0, "CmdGetBuffer32 failed: unexpected status 0x%02X ...(%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
REGISTERWRITE (0x0C, reg | 0x40);
|
|
|
|
read = 0;
|
|
while (read < len)
|
|
{
|
|
if (read + 1700 < len)
|
|
{
|
|
tmp = 1700;
|
|
EPPRead32Buffer (tmp, buffer + read);
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
if ((read + tmp < len) && (reg & 0x08) == 0x08)
|
|
{
|
|
do
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
while ((reg & 0x08) == 0x08);
|
|
if ((reg != 0xC0) && (reg != 0xD0))
|
|
{
|
|
DBG (0,
|
|
"Unexpected status 0x%02X, expected 0xC0 or 0xD0 ! (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Going on...\n");
|
|
}
|
|
}
|
|
reg = RegisterRead (0x0C);
|
|
RegisterWrite (0x0C, reg | 0x40);
|
|
read += tmp;
|
|
}
|
|
else
|
|
{
|
|
tmp = len - read;
|
|
EPPRead32Buffer (tmp, buffer + read);
|
|
read += tmp;
|
|
if ((read < len))
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
while ((reg & 0x08) == 0x08)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* OK ! */
|
|
Epilogue ();
|
|
return (1);
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_CmdSync (int cmd)
|
|
{
|
|
int word[5];
|
|
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdSync: Prologue failed ! (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
|
|
/* compute word */
|
|
word[0] = 0x00;
|
|
word[1] = 0x00;
|
|
word[2] = 0x00;
|
|
word[3] = cmd;
|
|
|
|
|
|
/* send data */
|
|
if (SendLength (word, 4) == 0)
|
|
{
|
|
DBG (0, "SendLength(word,4) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendLength(word,4) passed ...");
|
|
|
|
/* end OK */
|
|
Epilogue ();
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* numbers of bytes read, else 0 (failed) */
|
|
/* read data by chunk EXACTLY the width of the scan area in the given */
|
|
/* resolution . If a valid file descriptor is given, we write data */
|
|
/* in it according to the color mode, before polling the scanner */
|
|
/* len should not be bigger than 2 Megs */
|
|
|
|
int
|
|
CmdGetBlockBuffer (int cmd, int len, int window, unsigned char *buffer)
|
|
{
|
|
#ifdef HAVE_SYS_TIME_H
|
|
struct timeval td, tf;
|
|
float elapsed;
|
|
#endif
|
|
int reg, i;
|
|
int word[5], read;
|
|
|
|
/* compute word */
|
|
word[0] = len / 65536;
|
|
word[1] = len / 256 % 256;
|
|
word[2] = len % 256;
|
|
word[3] = (cmd & 0x3F) | 0x80 | 0x40;
|
|
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdGetBlockBuffer: Prologue failed ! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
/* send data */
|
|
if (SendLength (word, 4) == 0)
|
|
{
|
|
DBG (0, "SendLength(word,4) failed (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "SendLength(word,4) passed ...");
|
|
/* head end */
|
|
Epilogue ();
|
|
|
|
|
|
|
|
if (!Prologue ())
|
|
{
|
|
DBG (0, "CmdGetBlockBuffer: Prologue failed ! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
}
|
|
|
|
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
|
|
i = 0;
|
|
|
|
/* init counter */
|
|
read = 0;
|
|
|
|
/* read scanner state */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
|
|
|
|
/* read loop */
|
|
while (read < len)
|
|
{
|
|
/* wait for the data to be ready */
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&td, NULL);
|
|
#endif
|
|
while ((reg & 0x08) == 0x08)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&tf, NULL);
|
|
elapsed =
|
|
((tf.tv_sec * 1000000 + tf.tv_usec) -
|
|
(td.tv_sec * 1000000 + td.tv_usec)) / 1000000;
|
|
if (elapsed > 3)
|
|
{
|
|
DBG
|
|
(0,
|
|
"Time-out (%.2f s) waiting for scanner ... giving up on status 0x%02X ! (%s:%d)\n",
|
|
elapsed, reg, __FILE__, __LINE__);
|
|
Epilogue ();
|
|
return (read);
|
|
}
|
|
#endif
|
|
}
|
|
if ((reg != 0xC0) && (reg != 0xD0) && (reg != 0x00))
|
|
{
|
|
DBG (0,
|
|
"Unexpected status 0x%02X, expected 0xC0 or 0xD0 ! (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Going on...\n");
|
|
}
|
|
|
|
/* signals next chunk */
|
|
reg = RegisterRead (0x0C);
|
|
if (reg != 0x04)
|
|
{
|
|
DBG (0,
|
|
"CmdGetBlockBuffer failed: unexpected value reg0C=0x%02X ...(%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
REGISTERWRITE (0x0C, reg | 0x40);
|
|
|
|
|
|
/* there is always a full block ready when scanner is ready */
|
|
/* 32 bits I/O read , window must match the width of scan */
|
|
if (GetEPPMode () == 32)
|
|
EPPRead32Buffer (window, buffer + read);
|
|
else
|
|
EPPReadBuffer (window, buffer + read);
|
|
/* sum bytes read */
|
|
read += window;
|
|
|
|
|
|
DBG (16, "Read %d bytes out of %d (last block is %d bytes) (%s:%d)\n",
|
|
read, len, window, __FILE__, __LINE__);
|
|
|
|
/* test status after read */
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
}
|
|
|
|
|
|
/* wait for the data to be ready */
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&td, NULL);
|
|
#endif
|
|
while ((reg & 0x08) == 0x08)
|
|
{
|
|
reg = RegisterRead (0x19) & 0xF8;
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&tf, NULL);
|
|
elapsed =
|
|
((tf.tv_sec * 1000000 + tf.tv_usec) -
|
|
(td.tv_sec * 1000000 + td.tv_usec)) / 1000000;
|
|
if (elapsed > 3)
|
|
{
|
|
DBG
|
|
(0,
|
|
"Time-out (%.2f s) waiting for scanner ... giving up on status 0x%02X ! (%s:%d)\n",
|
|
elapsed, reg, __FILE__, __LINE__);
|
|
Epilogue ();
|
|
return (read);
|
|
}
|
|
#endif
|
|
}
|
|
if ((reg != 0xC0) && (reg != 0xD0) && (reg != 0x00))
|
|
{
|
|
DBG (0, "Unexpected status 0x%02X, expected 0xC0 or 0xD0 ! (%s:%d)\n",
|
|
reg, __FILE__, __LINE__);
|
|
DBG (0, "Going on...\n");
|
|
}
|
|
|
|
REGISTERWRITE (0x0E, 0x0D);
|
|
REGISTERWRITE (0x0F, 0x00);
|
|
|
|
|
|
/* OK ! */
|
|
Epilogue ();
|
|
return (read);
|
|
}
|
|
|
|
static void
|
|
Bloc2Decode (int *op)
|
|
{
|
|
int i;
|
|
int scanh;
|
|
int skiph;
|
|
int dpi = 0;
|
|
int dir = 0;
|
|
int color = 0;
|
|
char str[64];
|
|
|
|
for (i = 0; i < 16; i++)
|
|
sprintf (str + 3 * i, "%02X ", op[i]);
|
|
str[48] = 0x00;
|
|
DBG (0, "Command bloc 2: %s\n", str);
|
|
|
|
|
|
scanh = op[0] + (op[1] & 0x3F) * 256;
|
|
skiph = ((op[1] & 0xC0) >> 6) + (op[2] * 4) + ((op[3] & 0x0F) << 10);
|
|
|
|
if (op[3] & 0x10)
|
|
dir = 1;
|
|
else
|
|
dir = 0;
|
|
|
|
if (op[13] & 0x04)
|
|
color = 1;
|
|
else
|
|
color = 0;
|
|
|
|
/* op[6]=0x60 at 600 and 1200 dpi */
|
|
if ((op[8] == 0x17) && (op[9] != 0x05))
|
|
dpi = 150;
|
|
if ((op[8] == 0x17) && (op[9] == 0x05))
|
|
dpi = 300;
|
|
if ((op[9] == 0x05) && (op[14] & 0x08))
|
|
dpi = 1200;
|
|
if ((dpi == 0) && ((op[14] & 0x08) == 0))
|
|
dpi = 600;
|
|
|
|
|
|
|
|
DBG (0, "\t->scan height =0x%04X (%d)\n", scanh, scanh);
|
|
DBG (0, "\t->skip height =0x%04X (%d)\n", skiph, skiph);
|
|
DBG (0, "\t->y dpi =0x%04X (%d)\n", dpi, dpi);
|
|
DBG (0, "\t->channel 1 brightness=0x%02X (%d)\n", op[10] & 0x0F,
|
|
op[10] & 0x0F);
|
|
DBG (0, "\t->channel 2 brightness=0x%02X (%d)\n", (op[10] & 0xF0) / 16,
|
|
(op[10] & 0xF0) / 16);
|
|
DBG (0, "\t->channel 3 brightness=0x%02X (%d)\n", op[11] & 0x0F,
|
|
op[11] & 0x0F);
|
|
DBG (0, "\t->channel 1 high=0x%02X (%d)\n", (op[11] / 16) & 0x0F,
|
|
(op[11] / 16) & 0x0F);
|
|
DBG (0, "\t->channel 2 high=0x%02X (%d)\n", (op[12] & 0xF0) / 16,
|
|
(op[12] & 0xF0) / 16);
|
|
DBG (0, "\t->channel 3 high=0x%02X (%d)\n", op[12] & 0x0F, op[12] & 0x0F);
|
|
if (dir)
|
|
DBG (0, "\t->forward direction\n");
|
|
else
|
|
DBG (0, "\t->reverse direction\n");
|
|
if (color)
|
|
DBG (0, "\t->color scan \n");
|
|
else
|
|
DBG (0, "\t->no color scan \n");
|
|
|
|
/* byte 14 */
|
|
if (op[14] & 0x20)
|
|
{
|
|
DBG (0, "\t->lamp on \n");
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "\t->lamp off \n");
|
|
}
|
|
if (op[14] & 0x04)
|
|
{
|
|
DBG (0, "\t->normal scan (head stops at each row) \n");
|
|
}
|
|
else
|
|
{
|
|
DBG (0, "\t->move and scan (head doesn't stop at each row) \n");
|
|
}
|
|
DBG (0, "\n");
|
|
}
|
|
|
|
|
|
static void
|
|
Bloc8Decode (int *op)
|
|
{
|
|
int i, bpl;
|
|
int xskip;
|
|
int xend;
|
|
char str[128];
|
|
|
|
for (i = 0; i < 36; i++)
|
|
sprintf (str + 3 * i, "%02X ", op[i]);
|
|
str[3 * i] = 0x00;
|
|
DBG (0, "Command bloc 8: %s\n", str);
|
|
|
|
xskip = op[17] + 256 * (op[18] & 0x0F);
|
|
if (op[33] & 0x40)
|
|
xskip += 0x1000;
|
|
xend = (op[18] & 0xF0) / 16 + 16 * op[19];
|
|
if (op[33] & 0x80)
|
|
xend += 0x1000;
|
|
bpl = (op[24] - 0x41) * 256 + 8192 * (op[34] & 0x01) + op[23];
|
|
|
|
DBG (0, "\t->xskip =0x%X (%d)\n", xskip, xskip);
|
|
DBG (0, "\t->xend =0x%X (%d)\n", xend, xend);
|
|
DBG (0, "\t->scan width=0x%X (%d)\n", xend - xskip - 1, xend - xskip - 1);
|
|
DBG (0, "\t->bytes/line=0x%X (%d)\n", bpl, bpl);
|
|
DBG (0, "\n");
|
|
}
|
|
|
|
static int
|
|
CompletionWait (void)
|
|
{
|
|
CMDSYNC (0x40);
|
|
do
|
|
{
|
|
usleep (100000);
|
|
CMDSYNC (0xC2);
|
|
}
|
|
while ((sanei_umax_pp_ScannerStatus () & 0x90) != 0x90);
|
|
CMDSYNC (0xC2);
|
|
return (1);
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_SetLamp (int on)
|
|
{
|
|
int buffer[17];
|
|
int state;
|
|
|
|
|
|
|
|
/* reset? */
|
|
sanei_umax_pp_CmdSync (0x00);
|
|
sanei_umax_pp_CmdSync (0xC2);
|
|
sanei_umax_pp_CmdSync (0x00);
|
|
|
|
/* get status */
|
|
CmdGet (0x02, 16, buffer);
|
|
state = buffer[14] & LAMP_STATE;
|
|
buffer[16] = -1;
|
|
if ((state == 0) && (on == 0))
|
|
{
|
|
DBG (0, "Lamp already off ... (%s:%d)\n", __FILE__, __LINE__);
|
|
return (1);
|
|
}
|
|
if ((state) && (on))
|
|
{
|
|
DBG (2, "Lamp already on ... (%s:%d)\n", __FILE__, __LINE__);
|
|
return (1);
|
|
}
|
|
|
|
/* set lamp state */
|
|
if (on)
|
|
buffer[14] = buffer[14] | LAMP_STATE;
|
|
else
|
|
buffer[14] = buffer[14] & ~LAMP_STATE;
|
|
if (CmdSetGet (0x02, 16, buffer) != 1)
|
|
{
|
|
DBG (0, "CmdSetGet(0x02,16,buffer) failed (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
DBG (16, "CmdSetGet(0x02,16,buffer) passed ... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (1);
|
|
}
|
|
|
|
static int num = 0;
|
|
static void
|
|
Dump (int len, unsigned char *data, char *name)
|
|
{
|
|
FILE *fic;
|
|
char titre[80];
|
|
|
|
if (name == NULL)
|
|
{
|
|
sprintf (titre, "dump%04d.bin", num);
|
|
num++;
|
|
}
|
|
else
|
|
{
|
|
sprintf (titre, "%s", name);
|
|
}
|
|
fic = fopen (titre, "wb");
|
|
if (fic == NULL)
|
|
{
|
|
DBG (0, "could not open %s for writing\n", titre);
|
|
return;
|
|
}
|
|
fwrite (data, 1, len, fic);
|
|
fclose (fic);
|
|
}
|
|
|
|
|
|
static void
|
|
DumpNB (int width, int height, unsigned char *data, char *name)
|
|
{
|
|
FILE *fic;
|
|
char titre[80];
|
|
|
|
if (name == NULL)
|
|
{
|
|
sprintf (titre, "dump%04d.pnm", num);
|
|
num++;
|
|
}
|
|
else
|
|
{
|
|
sprintf (titre, "%s", name);
|
|
}
|
|
fic = fopen (titre, "wb");
|
|
if (fic == NULL)
|
|
{
|
|
DBG (0, "could not open %s for writing\n", titre);
|
|
return;
|
|
}
|
|
fprintf (fic, "P5\n%d %d\n255\n", width, height);
|
|
fwrite (data, width, height, fic);
|
|
fclose (fic);
|
|
}
|
|
|
|
|
|
/* dump data has received from scanner (red line/green line/blue line)
|
|
to a color pnm file */
|
|
static void
|
|
DumpRVB (int width, int height, unsigned char *data, char *name)
|
|
{
|
|
FILE *fic;
|
|
char titre[80];
|
|
int y, x;
|
|
|
|
if (name == NULL)
|
|
{
|
|
sprintf (titre, "dump%04d.pnm", num);
|
|
num++;
|
|
}
|
|
else
|
|
{
|
|
sprintf (titre, "%s", name);
|
|
}
|
|
fic = fopen (titre, "wb");
|
|
if (fic == NULL)
|
|
{
|
|
DBG (0, "could not open %s for writing\n", titre);
|
|
return;
|
|
}
|
|
fprintf (fic, "P6\n%d %d\n255\n", width, height);
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
fputc (data[3 * y * width + 2 * width + x], fic);
|
|
fputc (data[3 * y * width + width + x], fic);
|
|
fputc (data[3 * y * width + x], fic);
|
|
}
|
|
}
|
|
fclose (fic);
|
|
}
|
|
|
|
/* dump a color buffer in a color PNM */
|
|
static void
|
|
DumpRGB (int width, int height, unsigned char *data, char *name)
|
|
{
|
|
FILE *fic;
|
|
char titre[80];
|
|
int y, x;
|
|
|
|
if (name == NULL)
|
|
{
|
|
sprintf (titre, "dump%04d.pnm", num);
|
|
num++;
|
|
}
|
|
else
|
|
{
|
|
sprintf (titre, "%s", name);
|
|
}
|
|
fic = fopen (titre, "wb");
|
|
fprintf (fic, "P6\n%d %d\n255\n", width, height);
|
|
if (fic == NULL)
|
|
{
|
|
DBG (0, "could not open %s for writing\n", titre);
|
|
return;
|
|
}
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
fputc (data[3 * y * width + x * 3], fic);
|
|
fputc (data[3 * y * width + x * 3 + 1], fic);
|
|
fputc (data[3 * y * width + x * 3 + 2], fic);
|
|
}
|
|
}
|
|
fclose (fic);
|
|
}
|
|
|
|
static int
|
|
EvalGain (int sum, int count)
|
|
{
|
|
int gn;
|
|
float pct;
|
|
float avg;
|
|
|
|
|
|
/*if(getenv("CORRECTION"))
|
|
return(atoi(getenv("CORRECTION"))); */
|
|
|
|
/* 19000 is a little too bright */
|
|
/* gn = (int) ((double) (18500 * count) / sum - 100 + 0.5); */
|
|
|
|
/* after ~ 60 * 10 scans , it looks like 1 step is a 0.57% increase */
|
|
/* so we take the value and compute the percent increase to reach 250 */
|
|
/* not 255, because we want some room for inaccuracy */
|
|
/* pct=100-(value*100)/250 */
|
|
/* then correction is pct/0.57 */
|
|
avg = (float) (sum) / (float) (count);
|
|
pct = 100.0 - (avg * 100.0) / 250.0;
|
|
gn = (int) (pct / 0.57);
|
|
|
|
/* bound checking : there are sightings of >127 values being negative */
|
|
if (gn < 0)
|
|
gn = 0;
|
|
else if (gn > 127)
|
|
gn = 127;
|
|
return (gn);
|
|
}
|
|
|
|
static void
|
|
ComputeCalibrationData (int color, int dpi, int width, unsigned char *source,
|
|
int *data)
|
|
{
|
|
int p, i, l;
|
|
int sum;
|
|
|
|
|
|
memset (data, 0, (3 * 5100 + 768 + 3) * sizeof (int));
|
|
|
|
|
|
/* 0 -> 5099 */
|
|
for (i = 0; i < width; i++)
|
|
{ /* red calibration data */
|
|
if (color >= RGB_MODE)
|
|
{
|
|
/* compute average */
|
|
sum = 0;
|
|
for (l = 0; l < 66; l++)
|
|
sum += source[i + l * 5100 * 3];
|
|
data[i] = EvalGain (sum, l);
|
|
}
|
|
else
|
|
data[i] = 0x00;
|
|
}
|
|
|
|
|
|
/* 5100 -> 10199: green data */
|
|
p = 5100;
|
|
for (i = 0; i < width; i++)
|
|
{
|
|
/* compute average */
|
|
sum = 0;
|
|
for (l = 0; l < 66; l++)
|
|
sum += source[i + l * 5100 * 3 + 5100];
|
|
data[p + i] = EvalGain (sum, l);
|
|
}
|
|
|
|
|
|
/* 10200 -> 15299: blue */
|
|
p = 10200;
|
|
for (i = 0; i < width; i++)
|
|
{
|
|
if (color >= RGB_MODE)
|
|
{
|
|
/* compute average */
|
|
sum = 0;
|
|
for (l = 0; l < 66; l++)
|
|
sum += source[i + l * 5100 * 3 + 5100 * 2];
|
|
data[p + i] = EvalGain (sum, l);
|
|
}
|
|
else
|
|
data[p + i] = 0x00;
|
|
}
|
|
|
|
|
|
/* gamma tables */
|
|
for (i = 0; i < 256; i++)
|
|
data[15300 + i] = ggRed[i];
|
|
for (i = 0; i < 256; i++)
|
|
data[15300 + 256 + i] = ggGreen[i];
|
|
for (i = 0; i < 256; i++)
|
|
data[15300 + 512 + i] = ggBlue[i];
|
|
|
|
|
|
|
|
|
|
if (color >= RGB_MODE)
|
|
{
|
|
switch (dpi)
|
|
{
|
|
case 1200:
|
|
case 600:
|
|
data[16068] = 0xFF;
|
|
data[16069] = 0xFF;
|
|
break;
|
|
case 300:
|
|
data[16068] = 0xAA;
|
|
data[16069] = 0xFF;
|
|
break;
|
|
case 150:
|
|
data[16068] = 0x88;
|
|
data[16069] = 0xFF;
|
|
break;
|
|
case 75:
|
|
data[16068] = 0x80;
|
|
data[16069] = 0xAA;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (dpi)
|
|
{
|
|
case 1200:
|
|
case 600:
|
|
data[16068] = 0xFF;
|
|
data[16069] = 0xFF;
|
|
break;
|
|
case 300:
|
|
data[16068] = 0xAA;
|
|
data[16069] = 0xFF;
|
|
break;
|
|
case 150:
|
|
data[16068] = 0x88;
|
|
data[16069] = 0xAA;
|
|
break;
|
|
case 75:
|
|
data[16068] = 0x80;
|
|
data[16069] = 0x88;
|
|
break;
|
|
}
|
|
}
|
|
|
|
data[16070] = -1;
|
|
}
|
|
|
|
|
|
|
|
/* move head by the distance given using precision or not */
|
|
/* 0: failed
|
|
1: success */
|
|
static int
|
|
Move (int distance, int precision, unsigned char *buffer)
|
|
{
|
|
int header[17] =
|
|
{ 0x01, 0x00, 0x00, 0x20, 0x00, 0x00, 0x60, 0x2F, 0x2F, 0x01, 0x00, 0x00,
|
|
0x00, 0x80, 0xA4, 0x00, -1
|
|
};
|
|
int body[37] =
|
|
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0xF6, 0x79, 0xBF, 0x01, 0x00, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int end[9] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, -1 };
|
|
int steps, len;
|
|
unsigned char tmp[0x200];
|
|
unsigned char *ptr;
|
|
|
|
if (distance == 0)
|
|
return (0);
|
|
|
|
if (buffer == NULL)
|
|
ptr = tmp;
|
|
else
|
|
ptr = buffer;
|
|
|
|
/* build commands */
|
|
if (distance < 0)
|
|
{
|
|
/* header */
|
|
steps = -distance - 1;
|
|
header[3] = 0x20;
|
|
header[9] = 0x01;
|
|
|
|
/* reverse direction body by default */
|
|
|
|
/* end */
|
|
end[1] = 0xFF;
|
|
end[2] = 0xFF;
|
|
end[3] = -1;
|
|
len = 3;
|
|
}
|
|
else
|
|
{
|
|
/* header */
|
|
steps = distance - 1;
|
|
header[3] = 0x70;
|
|
header[9] = 0x05;
|
|
|
|
/* body */
|
|
body[2] = 0x04;
|
|
body[4] = 0x02;
|
|
body[7] = 0x0C;
|
|
body[9] = 0x04;
|
|
body[10] = 0x40;
|
|
body[11] = 0x01;
|
|
/* end */
|
|
len = 8;
|
|
}
|
|
if (steps > 0)
|
|
{
|
|
header[1] = (steps << 6) & 0xFF;
|
|
header[2] = (steps >> 2) & 0xFF;
|
|
header[3] = header[3] | ((steps >> 10) & 0x0F);
|
|
}
|
|
|
|
|
|
/* precision: default header set to precision on */
|
|
if (precision == PRECISION_OFF)
|
|
{
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
header[8] = 0x15;
|
|
else
|
|
header[8] = 0x17;
|
|
header[14] = 0xAC;
|
|
body[20] = 0x06;
|
|
}
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Bloc2Decode (header);
|
|
Bloc8Decode (body);
|
|
}
|
|
CMDSETGET (0x02, 16, header);
|
|
CMDSETGET (0x08, 36, body);
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Bloc2Decode (header);
|
|
Bloc8Decode (body);
|
|
}
|
|
CMDSYNC (0xC2);
|
|
if (sanei_umax_pp_ScannerStatus () & 0x80)
|
|
{
|
|
CMDSYNC (0x00);
|
|
}
|
|
CMDSETGET (4, len, end);
|
|
COMPLETIONWAIT;
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Dump (0x200, ptr, NULL);
|
|
}
|
|
CMDGETBUF (4, 0x200, ptr);
|
|
DBG (16, "MOVE STATUS IS 0x%02X (%s:%d)\n", sanei_umax_pp_ScannerStatus (),
|
|
__FILE__, __LINE__);
|
|
CMDSYNC (0x00);
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
/* for each column, finds the row where white/black transition occurs
|
|
then returns the average */
|
|
static float
|
|
EdgePosition (int width, int height, unsigned char *data)
|
|
{
|
|
int ecnt, x, y;
|
|
float epos;
|
|
int d, dmax, dpos, i;
|
|
unsigned char *dbuffer = NULL;
|
|
|
|
if (DBG_LEVEL > 128)
|
|
{
|
|
dbuffer = (unsigned char *) malloc (3 * width * height);
|
|
memset (dbuffer, 0x00, 3 * width * height);
|
|
}
|
|
epos = 0;
|
|
ecnt = 0;
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
/* edge: white->black drop */
|
|
/* loop stops on black area */
|
|
dmax = 0;
|
|
dpos = 0;
|
|
d = 0;
|
|
i = 0;
|
|
for (y = 10; (y < height) && (data[i] > 10); y++)
|
|
{
|
|
i = x + y * width;
|
|
d = data[i - width] - data[i];
|
|
if (d > dmax)
|
|
{
|
|
dmax = d;
|
|
dpos = y;
|
|
}
|
|
if ((DBG_LEVEL > 128) && (dbuffer != NULL))
|
|
{
|
|
dbuffer[i * 3] = data[i];
|
|
dbuffer[i * 3 + 1] = data[i];
|
|
dbuffer[i * 3 + 2] = data[i];
|
|
}
|
|
}
|
|
epos += dpos;
|
|
ecnt++;
|
|
if ((DBG_LEVEL > 128) && (dbuffer != NULL))
|
|
{
|
|
dbuffer[(x + dpos * width) * 3] = 0xFF;
|
|
dbuffer[(x + dpos * width) * 3 + 1] = 0x00;
|
|
dbuffer[(x + dpos * width) * 3 + 2] = 0x00;
|
|
}
|
|
}
|
|
if (ecnt == 0)
|
|
epos = 70;
|
|
else
|
|
epos = epos / ecnt;
|
|
if ((DBG_LEVEL > 128) && (dbuffer != NULL))
|
|
{
|
|
i = ((int) epos) * width;
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
dbuffer[(x + i) * 3] = 0x00;
|
|
dbuffer[(x + i) * 3 + 1] = 0xFF;
|
|
dbuffer[(x + i) * 3 + 2] = 0xFF;
|
|
}
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
dbuffer[(width / 2 + y * width) * 3] = 0x00;
|
|
dbuffer[(width / 2 + y * width) * 3 + 1] = 0xFF;
|
|
dbuffer[(width / 2 + y * width) * 3 + 2] = 0x00;
|
|
}
|
|
DumpRGB (width, height, dbuffer, NULL);
|
|
free (dbuffer);
|
|
}
|
|
return (epos);
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
MoveToOrigin (void)
|
|
{
|
|
unsigned char buffer[54000];
|
|
float edge;
|
|
int val, delta;
|
|
int header[17] =
|
|
{ 0xB4, 0x00, 0x00, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x2F, 0x05, 0x00, 0x00,
|
|
0x00, 0x80, 0xA4, 0x00, -1
|
|
};
|
|
int body[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0xFB, 0xC4, 0xE5, 0x06, 0x00, 0x00, 0x60,
|
|
0x4D, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int end[9] = { 0x06, 0xF4, 0xFF, 0x81, 0x1B, 0x00, 0x08, 0x00, -1 };
|
|
int opsc03[9] = { 0x00, 0x00, 0x00, 0xAA, 0xCC, 0xEE, 0x80, 0xFF, -1 };
|
|
|
|
/* 1600P command set */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
header[8] = 0x2B;
|
|
|
|
body[29] = 0x1A;
|
|
body[30] = 0xEE;
|
|
|
|
end[0] = 0x19;
|
|
end[1] = 0xD5;
|
|
end[4] = 0x1B;
|
|
}
|
|
|
|
/* sync scanner then move head by the estimated origin */
|
|
CMDSYNC (0x00);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
MOVE (196, PRECISION_OFF, NULL);
|
|
|
|
/* scan 2400 'x' by 180 y, B&W at 600 dpi, starting at the */
|
|
/* first quarter of the x area. This should cover a black */
|
|
/* line drawn in the middle of a white area, under the piece */
|
|
/* that separates the 'little' window and the scan window */
|
|
CMDSETGET (0x02, 16, header);
|
|
CMDSETGET (0x08, 36, body);
|
|
if (DBG_LEVEL > 128)
|
|
{
|
|
Bloc2Decode (header);
|
|
Bloc8Decode (body);
|
|
}
|
|
CMDSETGET (0x01, 8, end);
|
|
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
/* signals black & white ? */
|
|
CMDSETGET (0x04, 8, opsc03);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (0x04, 54000, buffer); /* get find position data */
|
|
if (DBG_LEVEL > 128)
|
|
{
|
|
DumpNB (300, 180, buffer, NULL);
|
|
}
|
|
|
|
/* detection of 1600P is a by product of origin finding */
|
|
edge = 0.0;
|
|
for (val = 0; val < 54000; val++)
|
|
if (buffer[val] > edge)
|
|
edge = buffer[val];
|
|
DBG (32, "MAX VALUE=%f (%s:%d)\n", edge, __FILE__, __LINE__);
|
|
if ((edge <= 30) && (sanei_umax_pp_getastra () != 1600))
|
|
{
|
|
DBG (2, "MoveToOrigin() detected a 1600P");
|
|
sanei_umax_pp_setastra (1600);
|
|
}
|
|
edge = EdgePosition (300, 180, buffer);
|
|
/* rounded to lowest integer, since upping origin might lead */
|
|
/* to bump in the other side if doing a full size preview */
|
|
val = (int) (edge);
|
|
|
|
/* edge is 60 dots (at 600 dpi) from origin */
|
|
/* origin=current pos - 180 + edge + 60 */
|
|
/* grumpf, there is an offset somewhere ... */
|
|
delta = -188 + val;
|
|
DBG (64, "Edge=%f, val=%d, delta=%d\n", edge, val, delta);
|
|
|
|
/* move back to y-coordinate origin */
|
|
MOVE (delta, PRECISION_ON, NULL);
|
|
|
|
/* head successfully set to the origin */
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* computes color offset and gain */
|
|
/* X is red
|
|
Y is blue
|
|
Z is green
|
|
|
|
returns if OK, else 0
|
|
*/
|
|
|
|
static int
|
|
WarmUp (int color, int *brightness)
|
|
{
|
|
unsigned char buffer[5300];
|
|
int i, val, min, max;
|
|
int opsc02[9] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, -1 };
|
|
int opsc04[9] = { 0x06, 0xF4, 0xFF, 0x81, 0x1B, 0x00, 0x00, 0x00, -1 };
|
|
int opsc10[9] = { 0x06, 0xF4, 0xFF, 0x81, 0x1B, 0x00, 0x08, 0x00, -1 };
|
|
int opsc18[17] =
|
|
{ 0x01, 0x00, 0x00, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x2F, 0x00, 0x88, 0x08,
|
|
0x00, 0x80, 0xA4, 0x00, -1
|
|
};
|
|
int opsc38[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0x18, 0x10, 0x03, 0x06, 0x00, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int opsc39[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0x41, 0x20, 0x24, 0x06, 0x00, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int opsc40[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0x41, 0x60, 0x4F, 0x06, 0x00, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x93, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int opsc51[17] =
|
|
{ 0x09, 0x00, 0x00, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x2F, 0x00, 0xA5, 0x09,
|
|
0x00, 0x40, 0xA4, 0x00, -1
|
|
};
|
|
int opsc48[17] =
|
|
{ 0x09, 0x00, 0x00, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x2F, 0x00, 0x00, 0x00,
|
|
0x00, 0x40, 0xA4, 0x00, -1
|
|
};
|
|
float offsetX, offsetY, offsetZ;
|
|
|
|
|
|
/* really dirty hack: somethig is buggy in BW mode */
|
|
/* we override mode with color until the bug is found */
|
|
color = RGB_MODE;
|
|
|
|
/* 1600P have a different CCD command block */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[0] = 0x19;
|
|
opsc04[1] = 0xD5;
|
|
opsc04[4] = 0x1B;
|
|
opsc04[7] = 0x20;
|
|
|
|
opsc10[0] = 0x19;
|
|
opsc10[1] = 0xD5;
|
|
opsc10[4] = 0x1B;
|
|
opsc10[7] = 0x20;
|
|
|
|
opsc48[8] = 0x2B;
|
|
opsc48[11] = 0x20;
|
|
opsc48[12] = 0x08;
|
|
opsc48[13] = 0x42;
|
|
}
|
|
|
|
/* block that repeats : another kind of move .... */
|
|
/* scan one line of 24 bytes at the right/left */
|
|
/* maybe black calibration since it is out of the */
|
|
/* scanning area */
|
|
/* color calibration: offset detection */
|
|
/* scan 24 pixels wide, 1 pixel height at 600 dpi */
|
|
/* without moving the head. Pixels are 'under the */
|
|
/* roof', so should be black */
|
|
if (color >= RGB_MODE)
|
|
{
|
|
CMDSETGET (2, 0x10, opsc48);
|
|
CMDSETGET (8, 0x24, opsc38);
|
|
CMDSETGET (1, 0x08, opsc04); /* scan std, no 'enhancing' */
|
|
CMDSYNC (0xC2);
|
|
if (sanei_umax_pp_ScannerStatus () & 0x80)
|
|
{
|
|
CMDSYNC (0x00);
|
|
}
|
|
CMDSETGET (4, 0x08, opsc02); /* commit ? */
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetX = (float) val / i;
|
|
|
|
|
|
CMDSYNC (0x00);
|
|
opsc04[7] = opsc04[7] | 0x10; /* enhanced ? */
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetX += (float) val / i;
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
|
|
/* block that repeats */
|
|
/* must be monochrome since hscan=1 */
|
|
opsc48[0] = 0x01;
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc48[12] = 0x0C;
|
|
opsc48[13] = 0x82;
|
|
}
|
|
else
|
|
{
|
|
opsc48[12] = 0x04;
|
|
opsc48[13] = 0x80;
|
|
}
|
|
CMDSETGET (2, 0x10, opsc48);
|
|
CMDSETGET (8, 0x24, opsc38);
|
|
opsc04[7] = opsc04[7] & 0x20;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
if (sanei_umax_pp_ScannerStatus () & 0x80)
|
|
{
|
|
CMDSYNC (0x00);
|
|
}
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetY = (float) val / i;
|
|
|
|
CMDSYNC (0x00);
|
|
opsc04[7] = opsc04[7] | 0x10; /* brightness ? */
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetY += (float) val / i;
|
|
}
|
|
|
|
/* block that repeats */
|
|
if (color < RGB_MODE)
|
|
{
|
|
opsc48[0] = 0x05; /* B&H height is 5 */
|
|
opsc48[13] = 0xC0; /* B&W mode */
|
|
}
|
|
else
|
|
{
|
|
opsc48[0] = 0x05; /* color height is 5 (+4 ?) */
|
|
opsc48[13] = 0xC1; /* some strange mode ? */
|
|
}
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
opsc48[13] = opsc48[13] | 0x02;
|
|
CMDSETGET (2, 0x10, opsc48);
|
|
CMDSETGET (8, 0x24, opsc38);
|
|
opsc04[7] = opsc04[7] & 0x20;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
if (sanei_umax_pp_ScannerStatus () & 0x80)
|
|
{
|
|
CMDSYNC (0x00);
|
|
}
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetZ = (float) val / i;
|
|
|
|
CMDSYNC (0x00);
|
|
opsc04[7] = opsc04[7] | 0x10;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x18, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x18, buffer, NULL);
|
|
val = 0;
|
|
for (i = 0; i < 24; i++)
|
|
val += buffer[i];
|
|
offsetZ += (float) val / i;
|
|
|
|
|
|
/***********************/
|
|
/* auto brightness computing */
|
|
/***********************/
|
|
|
|
/* color correction set to 53 05 */
|
|
/* for a start */
|
|
*brightness = 0x535;
|
|
CMDSETGET (2, 0x10, opsc18);
|
|
CMDSETGET (8, 0x24, opsc39);
|
|
opsc04[7] = opsc04[7] & 0x20;
|
|
opsc04[6] = 0x06; /* one channel brightness value */
|
|
CMDSETGET (1, 0x08, opsc10); /* was opsc04, extraneaous string */
|
|
/* that prevents using Move .... */
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x200, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x200, buffer, NULL);
|
|
|
|
|
|
/* auto correction of brightness levels */
|
|
/* first color component X */
|
|
opsc51[10] = (*brightness) / 16; /* channel 1 & 2 brightness */
|
|
opsc51[11] = (*brightness) % 16; /* channel 3 brightness */
|
|
if (color >= RGB_MODE)
|
|
{
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc51[11] |= 0x20;
|
|
opsc51[12] = 0x08;
|
|
opsc51[13] |= 0x02;
|
|
|
|
opsc04[7] |= 0x20;
|
|
}
|
|
CMDSETGET (2, 0x10, opsc51);
|
|
CMDSETGET (8, 0x24, opsc40);
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Bloc2Decode (opsc51);
|
|
Bloc8Decode (opsc40);
|
|
}
|
|
opsc04[6] = (*brightness) / 256;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x14B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
while ((opsc04[6] < 0x0F) && (max < 250))
|
|
{
|
|
CMDSYNC (0x00);
|
|
opsc04[6]++;
|
|
CMDSETGET (1, 0x000008, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x0014B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
}
|
|
|
|
*brightness = (*brightness & 0xFF) + 256 * (opsc04[6] - 1);
|
|
opsc51[10] = *brightness / 16; /* channel 1 & 2 brightness */
|
|
opsc51[11] = *brightness % 16; /* channel 3 brightness */
|
|
opsc51[0] = 0x01;
|
|
opsc51[13] = 0x80;
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc51[11] |= 0x20;
|
|
opsc51[12] = 0x08;
|
|
opsc51[13] |= 0x02;
|
|
|
|
opsc04[7] |= 0x20;
|
|
}
|
|
CMDSETGET (2, 0x10, opsc51);
|
|
CMDSETGET (8, 0x24, opsc40);
|
|
opsc04[6] = (*brightness & 0x00F);
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x14B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
|
|
while ((opsc04[6] < 0x0F) && (max < 250))
|
|
{
|
|
CMDSYNC (0x00);
|
|
opsc04[6]++;
|
|
CMDSETGET (1, 0x000008, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x0014B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
}
|
|
*brightness = (*brightness & 0xFF0) + (opsc04[6] - 1);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* component Z: B&W component */
|
|
opsc51[10] = *brightness / 16; /* channel 1 & 2 brightness */
|
|
opsc51[11] = *brightness % 16; /* channel 3 brightness */
|
|
if (color < RGB_MODE)
|
|
opsc51[0] = 0x01; /* in BW, scan zone doesn't have an extra 4 points */
|
|
else
|
|
opsc51[0] = 0x05; /* extra 4 points */
|
|
opsc51[13] = 0xC0; /* B&W */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc51[11] |= 0x20;
|
|
opsc51[12] = 0x08;
|
|
opsc51[13] |= 0x02;
|
|
|
|
opsc04[7] |= 0x20;
|
|
}
|
|
CMDSETGET (2, 0x10, opsc51);
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Bloc2Decode (opsc51);
|
|
}
|
|
CMDSETGET (8, 0x24, opsc40);
|
|
opsc04[6] = (*brightness & 0x0F0) / 16;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (4, 0x08, opsc02);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x14B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
while ((opsc04[6] < 0x07) && (max < 250))
|
|
{
|
|
CMDSYNC (0x00);
|
|
opsc04[6]++;
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
COMPLETIONWAIT;
|
|
CMDGETBUF (4, 0x0014B4, buffer);
|
|
if (DBG_LEVEL >= 128)
|
|
Dump (0x14B4, buffer, NULL);
|
|
min = 255;
|
|
max = 0;
|
|
for (i = 0; i < 0x14B4; i++)
|
|
{
|
|
if (buffer[i] < min)
|
|
min = buffer[i];
|
|
if (buffer[i] > max)
|
|
max = buffer[i];
|
|
}
|
|
}
|
|
*brightness = (*brightness & 0xF0F) + (opsc04[6] - 1) * 16;
|
|
DBG (1, "Warm-up done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
/* park head: returns 1 on success, 0 otherwise */
|
|
/* distance is in 75 dpi unit */
|
|
int
|
|
sanei_umax_pp_Park (void)
|
|
{
|
|
int header[17] =
|
|
{ 0x01, 0x00, 0x01, 0x70, 0x00, 0x00, 0x60, 0x2F, 0x13, 0x05, 0x00, 0x00,
|
|
0x00, 0x80, 0xF0, 0x00, -1
|
|
};
|
|
int body[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x03, 0xC1, 0x80,
|
|
0x00, 0x00, 0x04, 0x00, 0x16, 0x80, 0x15, 0x78, 0x03, 0x03, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x1B, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int status = 0x90;
|
|
|
|
CMDSYNC (0x00);
|
|
|
|
CMDSETGET (0x02, 16, header);
|
|
if (DBG_LEVEL >= 32)
|
|
Bloc8Decode (body);
|
|
CMDSETGET (0x08, 36, body);
|
|
CMDSYNC (0x40);
|
|
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
DBG (16, "PARKING STATUS is 0x%02X (%s:%d)\n", status, __FILE__, __LINE__);
|
|
DBG (1, "Park command issued ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* calibrates CCD: returns 1 on success, 0 on failure */
|
|
static int
|
|
ColorCalibration (int color, int dpi, int brightness, int contrast, int width,
|
|
int *calibration)
|
|
{
|
|
int opsc32[17] =
|
|
{ 0x4A, 0x00, 0x00, 0x70, 0x00, 0x00, 0x60, 0x00, 0x17, 0x05, 0xA5, 0x08,
|
|
0x00, 0x00, 0xAC, 0x00, -1
|
|
};
|
|
int opsc41[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0x90, 0xD0, 0x47, 0x06, 0x00, 0x00, 0xC4,
|
|
0x5C, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x93, 0x1B, 0x00,
|
|
-1
|
|
};
|
|
int opscnb[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x04, 0x40, 0x01,
|
|
0x00, 0x00, 0x04, 0x00, 0x6E, 0x90, 0xD0, 0x47, 0x06, 0x00, 0x00, 0xEC,
|
|
0x54, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x93, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int opsc04[9] = { 0x06, 0xF4, 0xFF, 0x81, 0x1B, 0x00, 0x00, 0x00, -1 };
|
|
int opsc02[9] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, -1 };
|
|
int size;
|
|
unsigned char buffer[0x105798];
|
|
|
|
/* 1600P have a different CCD command block */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[0] = 0x19;
|
|
opsc04[1] = 0xD5;
|
|
opsc04[4] = 0x1B;
|
|
|
|
opsc41[29] = 0x1A;
|
|
opsc41[30] = 0xEE;
|
|
}
|
|
|
|
/* step back by 67 ticks: */
|
|
/* since we're going to scan 66 lines of data */
|
|
/* which are going to be used as calibration */
|
|
/* data */
|
|
/* we are on the white area just before */
|
|
/* the user scan area */
|
|
MOVE (-67, PRECISION_ON, NULL);
|
|
|
|
|
|
CMDSYNC (0x00);
|
|
|
|
/* get calibration data */
|
|
if (sanei_umax_pp_getauto ())
|
|
{ /* auto settings doesn't use contrast */
|
|
contrast = 0x000;
|
|
}
|
|
else
|
|
{ /* manual settings */
|
|
brightness = 0x777;
|
|
contrast = 0x000;
|
|
}
|
|
opsc32[10] = brightness / 16;
|
|
opsc32[11] = brightness % 16 | ((contrast / 16) & 0xF0);
|
|
opsc32[12] = contrast % 256;
|
|
DBG (8, "USING 0x%03X brightness, 0x%03X contrast\n", brightness, contrast);
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc32[13] = 0x03;
|
|
}
|
|
|
|
|
|
if (color < RGB_MODE)
|
|
{
|
|
opsc32[0] -= 4;
|
|
opsc32[13] = 0xC0;
|
|
}
|
|
CMDSETGET (2, 0x10, opsc32);
|
|
if (DBG_LEVEL >= 64)
|
|
{
|
|
Bloc2Decode (opsc32);
|
|
}
|
|
if (color < RGB_MODE)
|
|
{
|
|
CMDSETGET (8, 0x24, opscnb);
|
|
if (DBG_LEVEL >= 64)
|
|
{
|
|
Bloc8Decode (opscnb);
|
|
}
|
|
opsc04[6] = 0x85;
|
|
}
|
|
else
|
|
{
|
|
CMDSETGET (8, 0x24, opsc41);
|
|
if (DBG_LEVEL >= 64)
|
|
{
|
|
Bloc8Decode (opsc41);
|
|
}
|
|
opsc04[6] = 0x0F;
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
opsc04[7] = 0xC0;
|
|
else
|
|
opsc04[7] = 0x70;
|
|
}
|
|
|
|
/* BUG BW noisy here */
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
CMDSETGET (4, 0x08, opsc02); /* opsc03 hangs it */
|
|
COMPLETIONWAIT;
|
|
|
|
opsc04[0] = 0x06;
|
|
if (color >= RGB_MODE)
|
|
size = 3 * 5100 * 70;
|
|
else
|
|
size = 5100 * 66;
|
|
if (GetEPPMode () == 32)
|
|
{
|
|
CmdGetBuffer32 (4, size, buffer);
|
|
}
|
|
else
|
|
{
|
|
CMDGETBUF (4, size, buffer);
|
|
}
|
|
if (DBG_LEVEL >= 128)
|
|
{
|
|
Dump (size, buffer, NULL);
|
|
if (color >= RGB_MODE)
|
|
{
|
|
DumpRVB (5100, 66, buffer, NULL);
|
|
}
|
|
else
|
|
{
|
|
DumpNB (5100, 66, buffer, NULL);
|
|
}
|
|
}
|
|
ComputeCalibrationData (color, dpi, width, buffer, calibration);
|
|
|
|
DBG (1, "Color calibration done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
/* returns number of bytes read or 0 on failure */
|
|
int
|
|
sanei_umax_pp_ReadBlock (long len, int window, int dpi, int last,
|
|
unsigned char *buffer)
|
|
{
|
|
DBG (8, "ReadBlock(%ld,%d,%d,%d)\n", len, window, dpi, last);
|
|
/* EPP block reading is available only when dpi >=600 */
|
|
if (dpi >= 600)
|
|
{
|
|
DBG (8, "CmdGetBlockBuffer(4,%ld,%d);\n", len, window);
|
|
len = CmdGetBlockBuffer (4, len, window, buffer);
|
|
if (len == 0)
|
|
{
|
|
DBG (0, "CmdGetBlockBuffer(4,%ld,%d) failed (%s:%d)\n", len, window,
|
|
__FILE__, __LINE__);
|
|
gCancel = 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DBG (8, "CmdGetBuffer(4,%ld);\n", len);
|
|
if (CmdGetBuffer (4, len, buffer) != 1)
|
|
{
|
|
DBG (0, "CmdGetBuffer(4,%ld) failed (%s:%d)\n", len, __FILE__,
|
|
__LINE__);
|
|
gCancel = 1;
|
|
}
|
|
}
|
|
if (!last)
|
|
{
|
|
/* sync with scanner */
|
|
if (sanei_umax_pp_CmdSync (0xC2) == 0)
|
|
{
|
|
DBG (0, "Warning CmdSync(0xC2) failed! (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
DBG (0, "Trying again ... ");
|
|
if (sanei_umax_pp_CmdSync (0xC2) == 0)
|
|
{
|
|
DBG (0, " failed again! (%s:%d)\n", __FILE__, __LINE__);
|
|
DBG (0, "Aborting ...\n");
|
|
gCancel = 1;
|
|
}
|
|
else
|
|
DBG (0, " success ...\n");
|
|
|
|
}
|
|
}
|
|
return (len);
|
|
}
|
|
|
|
int
|
|
sanei_umax_pp_Scan (int x, int y, int width, int height, int dpi, int color,
|
|
int brightness, int contrast)
|
|
{
|
|
#ifdef HAVE_SYS_TIME_H
|
|
struct timeval td, tf;
|
|
float elapsed;
|
|
#endif
|
|
unsigned char *buffer;
|
|
long int somme, len, read, blocksize;
|
|
FILE *fout = NULL;
|
|
int *dest = NULL;
|
|
int bpl, hp;
|
|
int th, tw, bpp;
|
|
int distance = 0, nb;
|
|
int bx, by;
|
|
|
|
if (sanei_umax_pp_StartScan
|
|
(x, y, width, height, dpi, color, brightness, contrast, &bpp, &tw,
|
|
&th) == 1)
|
|
{
|
|
COMPLETIONWAIT;
|
|
|
|
/* blocksize must be multiple of the number of bytes per line */
|
|
/* max is 2096100 */
|
|
/* so blocksize will hold a round number of lines, easing the */
|
|
/* write data to file operation */
|
|
/*blocksize=(2096100/bpl)*bpl; */
|
|
bpl = bpp * tw;
|
|
hp = 16776960 / bpl; /* 16 Mo buffer (!!) */
|
|
hp = 2096100 / bpl;
|
|
blocksize = hp * bpl;
|
|
nb = 0;
|
|
read = 0;
|
|
|
|
/* get scanned data */
|
|
somme = bpp * tw * th;
|
|
DBG (8, "Getting buffer %d*%d*%d=%ld=0x%lX (%s:%d) \n", bpp, tw, th,
|
|
somme, somme, __FILE__, __LINE__);
|
|
|
|
/* allocate memory */
|
|
buffer = (unsigned char *) malloc (blocksize);
|
|
if (buffer == NULL)
|
|
{
|
|
DBG (0, "Failed to allocate %ld bytes, giving up....\n", blocksize);
|
|
DBG (0, "Try to scan at lower resolution, or a smaller area.\n");
|
|
gCancel = 1;
|
|
}
|
|
|
|
/* open output file */
|
|
fout = fopen ("out.pnm", "wb");
|
|
if (fout == NULL)
|
|
{
|
|
DBG (0, "Failed to open 'out.pnm', giving up....\n");
|
|
gCancel = 1;
|
|
}
|
|
else
|
|
{
|
|
/* write pnm header */
|
|
if (color >= RGB_MODE)
|
|
fprintf (fout, "P6\n%d %d\n255\n", tw, th);
|
|
else
|
|
fprintf (fout, "P5\n%d %d\n255\n", tw, th);
|
|
}
|
|
|
|
/* data reading loop */
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&td, NULL);
|
|
#endif
|
|
while ((read < somme) && (!gCancel))
|
|
{
|
|
/* 2096100 max */
|
|
if (somme - read > blocksize)
|
|
len = blocksize;
|
|
else
|
|
len = somme - read;
|
|
len =
|
|
sanei_umax_pp_ReadBlock (len, tw, dpi, (len < blocksize), buffer);
|
|
if (len == 0)
|
|
{
|
|
DBG (0, "ReadBlock failed, cancelling scan ...\n");
|
|
gCancel = 1;
|
|
}
|
|
|
|
read += len;
|
|
nb++;
|
|
DBG (8, "Read %ld bytes out of %ld ...\n", read, somme);
|
|
DBG (8, "Read %d blocks ... \n", nb);
|
|
|
|
/* write partial buffer to file */
|
|
if (len)
|
|
{
|
|
if (color >= RGB_MODE)
|
|
{
|
|
/* using an image format that doesn't need */
|
|
/* reordering would speed up write operation */
|
|
hp = len / bpl;
|
|
if (sanei_umax_pp_getastra () != 1600)
|
|
{
|
|
for (by = 0; by < hp; by++)
|
|
{
|
|
for (bx = 0; bx < tw; bx++)
|
|
{
|
|
fputc (buffer[3 * by * tw + 2 * tw + bx], fout);
|
|
fputc (buffer[3 * by * tw + tw + bx], fout);
|
|
fputc (buffer[3 * by * tw + bx], fout);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (by = 0; by < hp; by++)
|
|
{
|
|
for (bx = 0; bx < tw; bx++)
|
|
{
|
|
fputc (buffer[3 * by * tw + 2 * tw + bx], fout);
|
|
fputc (buffer[3 * by * tw + bx], fout);
|
|
fputc (buffer[3 * by * tw + tw + bx], fout);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
fwrite (buffer, len, 1, fout);
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_SYS_TIME_H
|
|
gettimeofday (&tf, NULL);
|
|
|
|
/* scan time are high enough to forget about usec */
|
|
elapsed = tf.tv_sec - td.tv_sec;
|
|
DBG (8, "%ld bytes transfered in %f seconds ( %.2f Kb/s)\n", somme,
|
|
elapsed, (somme / elapsed) / 1024.0);
|
|
#endif
|
|
|
|
/* release ressources */
|
|
if (fout != NULL)
|
|
fclose (fout);
|
|
free (dest);
|
|
free (buffer);
|
|
|
|
|
|
/* park head */
|
|
distance = distance + y + height;
|
|
} /* if start scan OK */
|
|
else
|
|
{
|
|
DBG (0, "StartScan failed..... \n");
|
|
}
|
|
|
|
/* terminate any pending command */
|
|
if (sanei_umax_pp_CmdSync (0x00) == 0)
|
|
{
|
|
DBG (0, "Warning CmdSync(0x00) failed! (%s:%d)\n", __FILE__, __LINE__);
|
|
DBG (0, "Trying again ... ");
|
|
if (sanei_umax_pp_CmdSync (0x00) == 0)
|
|
{
|
|
DBG (0, " failed again! (%s:%d)\n", __FILE__, __LINE__);
|
|
DBG (0, "Blindly going on ...\n");
|
|
}
|
|
else
|
|
DBG (0, " success ...\n");
|
|
|
|
}
|
|
|
|
/* parking */
|
|
if (sanei_umax_pp_Park () == 0)
|
|
DBG (0, "Park failed !!! (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
|
|
/* end ... */
|
|
DBG (1, "Scan done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
sanei_umax_pp_ParkWait (void)
|
|
{
|
|
int status;
|
|
|
|
/* check if head is at home */
|
|
do
|
|
{
|
|
sleep (2);
|
|
sanei_umax_pp_CmdSync (0x40);
|
|
status = sanei_umax_pp_ScannerStatus ();
|
|
}
|
|
while ((status & MOTOR_BIT) == 0x00);
|
|
DBG (1, "ParkWait done ...\n");
|
|
return (1);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* starts scan: return 1 on success */
|
|
int
|
|
sanei_umax_pp_StartScan (int x, int y, int width, int height, int dpi,
|
|
int color, int brightness, int contrast, int *rbpp,
|
|
int *rtw, int *rth)
|
|
{
|
|
unsigned char *buffer;
|
|
int *dest = NULL;
|
|
int state[16];
|
|
int err = 0;
|
|
int calibration[3 * 5100 + 768 + 2 + 1];
|
|
int xdpi, ydpi, bpl, h;
|
|
int th, tw, bpp;
|
|
int distance, i;
|
|
|
|
int opsc04[9] = { 0x06, 0xF4, 0xFF, 0x81, 0x1B, 0x00, 0x00, 0x00, -1 };
|
|
int opsc53[17] =
|
|
{ 0xA4, 0x80, 0x07, 0x50, 0xEC, 0x03, 0x00, 0x2F, 0x17, 0x07, 0x84, 0x08,
|
|
0x00, 0x00, 0xAC, 0x00, -1
|
|
};
|
|
int opsc35[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x03, 0xC1, 0x80,
|
|
0x00, 0x00, 0x04, 0x00, 0x16, 0x41, 0xE0, 0xAC, 0x03, 0x03, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
int opscan[37] =
|
|
{ 0x00, 0x00, 0xB0, 0x4F, 0xD8, 0xE7, 0xFA, 0x10, 0xEF, 0xC4, 0x3C, 0x71,
|
|
0x0F, 0x00, 0x04, 0x00, 0x6E, 0x61, 0xA1, 0x24, 0xC4, 0x7E, 0x00, 0xAE,
|
|
0x41, 0xA0, 0x0A, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x33, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
|
|
|
|
#ifdef UMAX_PP_DANGEROUS_EXPERIMENT
|
|
FILE *f = NULL;
|
|
char line[1024], *ptr;
|
|
int *base = NULL;
|
|
int channel;
|
|
int max = 0;
|
|
#endif
|
|
|
|
|
|
DBG (8, "StartScan(%d,%d,%d,%d,%d,%d,%X);\n", x, y, width, height, dpi,
|
|
color, brightness);
|
|
buffer = (unsigned char *) malloc (2096100);
|
|
if (buffer == NULL)
|
|
{
|
|
DBG (0, "Failed to allocate 2096100 bytes... (%s:%d)\n", __FILE__,
|
|
__LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* 1600P have a different CCD command block */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[0] = 0x19;
|
|
opsc04[1] = 0xD5;
|
|
opsc04[4] = 0x1B;
|
|
opsc04[7] = 0x70;
|
|
|
|
opscan[29] = 0x1A;
|
|
opscan[30] = 0xEE;
|
|
|
|
opsc35[29] = 0x1A;
|
|
opsc35[30] = 0xEE;
|
|
|
|
opsc53[13] = 0x03; /* may be blur filter */
|
|
}
|
|
|
|
/* get scanner status */
|
|
if (DBG_LEVEL > 8)
|
|
{
|
|
char str[64];
|
|
CMDGET (0x02, 16, state);
|
|
for (i = 0; i < 16; i++)
|
|
sprintf (str + 3 * i, "%02X ", state[i]);
|
|
str[48] = 0x00;
|
|
DBG (8, "SCANNER STATE=%s\n", str);
|
|
}
|
|
|
|
dest = (int *) malloc (65536 * 4);
|
|
if (sanei_umax_pp_getastra () != 1600)
|
|
{
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
|
|
if (dest == NULL)
|
|
{
|
|
DBG (0, "%s:%d failed to allocate 256 Ko !\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* init some buffer : default calibration data ? */
|
|
/* looks like a standard gamma table */
|
|
dest[0] = 0x00;
|
|
dest[1] = 0x00;
|
|
dest[2] = 0x00;
|
|
for (i = 0; i < 768; i++)
|
|
dest[i + 3] = i % 256;
|
|
dest[768 + 3] = 0xAA;
|
|
dest[768 + 4] = 0xAA;
|
|
dest[768 + 5] = -1;
|
|
CMDSETGET (0x04, 768 + 5, dest);
|
|
|
|
|
|
/* check buffer returned */
|
|
for (i = 0; i < 768; i++)
|
|
{
|
|
if (dest[i + 3] != (i % 256))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Error data altered: byte %d=0x%02X, should be 0x%02X ! (%s:%d)\n",
|
|
i, dest[i + 3], i % 256, __FILE__, __LINE__);
|
|
err = 1;
|
|
}
|
|
}
|
|
if (err)
|
|
return (0);
|
|
}
|
|
|
|
|
|
/* new part of buffer ... */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
dest[i * 2] = i;
|
|
dest[i * 2 + 1] = 0;
|
|
}
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
CMDSET (0x04, 512, dest);
|
|
|
|
/* another new part ... */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
dest[i * 2] = i;
|
|
dest[i * 2 + 1] = 0x04; /* instead of 0x00 */
|
|
}
|
|
opsc35[2] = 0x06; /* instead of 0x04, write flag ? */
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
CMDSET (0x04, 512, dest);
|
|
|
|
opsc35[2] = 0x04; /* return to initial value, read flag? */
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDGET (0x04, 512, dest);
|
|
|
|
/* check buffer returned */
|
|
/* if 0x4 are still 0x0 (hum..), we got a 1220P, else it is a 2000P */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
if ((dest[2 * i] != i)
|
|
|| ((dest[2 * i + 1] != 0x04) && (dest[2 * i + 1] != 0x00)))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Error data altered: expected %d=(0x%02X,0x04), found (0x%02X,0x%02X) ! (%s:%d)\n",
|
|
i, i, dest[i * 2], dest[i * 2 + 1], __FILE__, __LINE__);
|
|
err = 1;
|
|
}
|
|
}
|
|
if (err)
|
|
return (0);
|
|
|
|
/* find and move to zero */
|
|
if (MoveToOrigin () == 0)
|
|
{
|
|
DBG (0, "MoveToOrign() failed ... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
else
|
|
{
|
|
DBG (16, "MoveToOrign() passed ... (%s:%d)\n", __FILE__, __LINE__);
|
|
}
|
|
|
|
/* 1600P have a different CCD command block */
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[0] = 0x19;
|
|
opsc04[1] = 0xD5;
|
|
opsc04[4] = 0x1B;
|
|
opsc04[7] = 0x70;
|
|
|
|
opscan[29] = 0x1A;
|
|
opscan[30] = 0xEE;
|
|
|
|
opsc35[29] = 0x1A;
|
|
opsc35[30] = 0xEE;
|
|
|
|
opsc53[13] = 0x03; /* may be blur filter */
|
|
}
|
|
|
|
|
|
/* adjust brightness and color offset */
|
|
/* red*256+green*16+blue */
|
|
if (sanei_umax_pp_getauto ())
|
|
{
|
|
if (WarmUp (color, &brightness) == 0)
|
|
{
|
|
DBG (0, "Warm-up failed !!! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
/* x dpi is from 75 to 600 max, any modes */
|
|
if (dpi > 600)
|
|
xdpi = 600;
|
|
else
|
|
xdpi = dpi;
|
|
|
|
|
|
/* havent't yet found a way to make EPPRead32Buffer work */
|
|
/* with length not multiple of four bytes, so we enlarge */
|
|
/* width to meet this criteria ... */
|
|
if ((GetEPPMode () == 32) && (xdpi >= 600) && (width & 0x03))
|
|
{
|
|
width += (4 - (width & 0x03));
|
|
/* in case we go too far on the right */
|
|
if (x + width > 5100)
|
|
{
|
|
x = 5100 - width;
|
|
}
|
|
}
|
|
|
|
/* compute target size */
|
|
th = (height * dpi) / 600;
|
|
tw = (width * xdpi) / 600;
|
|
|
|
/* do gamma calibration */
|
|
if (ColorCalibration (color, dpi, brightness, contrast, width, calibration)
|
|
== 0)
|
|
{
|
|
DBG (0, "Gamma calibration failed !!! (%s:%d)\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
TRACE (16, "ColorCalibration() passed ...")
|
|
/* it is faster to move at low resolution, then scan */
|
|
/* than scan & move at high resolution */
|
|
distance = 0;
|
|
|
|
/* work around some strange unresolved bug */
|
|
y += 8;
|
|
|
|
/* move fast to scan target if possible */
|
|
if ((dpi > 150) && (y > 100))
|
|
{
|
|
distance = y;
|
|
|
|
/* move at 150 dpi resolution */
|
|
Move (y / 2, PRECISION_OFF, NULL);
|
|
|
|
/* keep the remainder for scan */
|
|
y = y % 4;
|
|
}
|
|
|
|
/* build final scan command */
|
|
|
|
/* round width and height */
|
|
width = (tw * 600) / xdpi;
|
|
height = (th * 600) / dpi;
|
|
|
|
ydpi = dpi;
|
|
if (color >= RGB_MODE)
|
|
{
|
|
if (dpi < 150)
|
|
ydpi = 150;
|
|
}
|
|
else
|
|
{
|
|
if (dpi < 300)
|
|
ydpi = 300;
|
|
}
|
|
|
|
if (color >= RGB_MODE)
|
|
{
|
|
h = ((height * ydpi) / 600) + 8;
|
|
bpp = 3;
|
|
}
|
|
else
|
|
{
|
|
h = ((height * ydpi) / 600) + 4;
|
|
bpp = 1;
|
|
}
|
|
bpl = (bpp * width * xdpi) / 600;
|
|
|
|
/* sets y resolution */
|
|
switch (ydpi)
|
|
{
|
|
case 1200:
|
|
opsc53[6] = 0x60;
|
|
opsc53[8] = 0x5E; /* *WORKING* value */
|
|
opsc53[8] = 0x5F; /* 5F gives wrong colors ? */
|
|
opsc53[8] = 0x58;
|
|
opsc53[9] = 0x05;
|
|
/* XXX test value XXX opsc53[14] = opsc53[14] & 0xF0; ~ 0x08 -> scan AND move */
|
|
/* XXX test value XXX opsc53[14] = (opsc53[14] & 0xF0) | 0x04; -> 600 dpi ? */
|
|
/* XXX test value XXX opsc53[14] = (opsc53[14] & 0xF0) | 0x0C; */
|
|
opsc53[14] = opsc53[14] & 0xF0; /* *WORKING* 1200 dpi */
|
|
break;
|
|
|
|
case 600:
|
|
opsc53[6] = 0x60;
|
|
opsc53[8] = 0x2F;
|
|
opsc53[9] = 0x05;
|
|
opsc53[14] = (opsc53[14] & 0xF0) | 0x04;
|
|
break;
|
|
|
|
case 300:
|
|
opsc53[6] = 0x00;
|
|
opsc53[8] = 0x17;
|
|
opsc53[9] = 0x05;
|
|
opsc53[14] = (opsc53[14] & 0xF0) | 0x0C;
|
|
|
|
/* si | 0C h=2*w, si | 04 h=w ? */
|
|
|
|
break;
|
|
|
|
case 150:
|
|
opsc53[6] = 0x00;
|
|
opsc53[8] = 0x17;
|
|
opsc53[9] = 0x07;
|
|
opsc53[14] = (opsc53[14] & 0xF0) | 0x0C;
|
|
break;
|
|
}
|
|
|
|
/* channels brightness */
|
|
opsc53[10] = brightness / 16;
|
|
opsc53[11] = ((contrast / 16) & 0xF0) | (brightness % 16);
|
|
opsc53[12] = contrast % 256;
|
|
|
|
/* scan height */
|
|
opsc53[0] = h % 256;
|
|
opsc53[1] = h / 256;
|
|
|
|
/* y start -1 */
|
|
y = (y * ydpi) / 600 - 1;
|
|
if (y > 0)
|
|
{
|
|
opsc53[1] |= (y % 4) * 64;
|
|
opsc53[2] = (y / 4) % 256;
|
|
opsc53[3] = 0x50 | ((y >> 10) & 0x0F);
|
|
}
|
|
else
|
|
{
|
|
opsc53[2] = 0x00;
|
|
opsc53[3] = 0x50;
|
|
}
|
|
|
|
/* x start -1 */
|
|
opscan[17] = (x - 1) % 256;
|
|
opscan[18] = (((x - 1) / 256) & 0x0F);
|
|
opscan[33] = 0x33;
|
|
if (x - 1 > 0x1000)
|
|
opscan[33] |= 0x40;
|
|
|
|
/* x end */
|
|
opscan[18] |= (((x + width) % 16) * 16);
|
|
opscan[19] = ((x + width) / 16) % 256;
|
|
if (x + width > 0x1000)
|
|
opscan[33] |= 0x80;
|
|
|
|
/* bytes per line */
|
|
opscan[24] = 0x41 + (bpl & 0x1FFF) / 256;
|
|
opscan[23] = bpl % 256;
|
|
|
|
|
|
if (color >= RGB_MODE)
|
|
{
|
|
opsc53[7] = 0x2F;
|
|
/* 00 seems to give better results ? */
|
|
/* 80 some more brightness, lamp power level ? */
|
|
/* 8x does not make much difference */
|
|
opsc04[6] = 0x8F;
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[7] = 0x70;
|
|
opsc53[13] = 0x03;
|
|
}
|
|
else
|
|
{
|
|
opsc04[7] = 0xF0;
|
|
opsc53[13] = 0x09;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
opsc53[7] = 0x40;
|
|
opsc53[13] = 0xC0;
|
|
opsc04[6] = 0x80 | ((brightness / 16) & 0x0F);
|
|
if (sanei_umax_pp_getastra () == 1600)
|
|
{
|
|
opsc04[7] = 0x20;
|
|
opsc53[13] = 0xC3;
|
|
}
|
|
else
|
|
{
|
|
opsc04[7] = 0xA0;
|
|
opsc53[13] = 0xC9;
|
|
}
|
|
}
|
|
|
|
|
|
CMDSYNC (0x00);
|
|
#ifdef UMAX_PP_DANGEROUS_EXPERIMENT
|
|
/*opsc53[13] = 0x80; B&W bit */
|
|
/*opsc53[13] = 0x40; green bit */
|
|
/*opsc53[13] = 0x20; red bit */
|
|
/*opsc53[13] = 0x10; blue bit */
|
|
/* with cmd 01, may be use to do 3 pass scanning ? */
|
|
/* bits 0 to 3 seem related to sharpness */
|
|
f = fopen ("/tmp/dangerous.params", "rb");
|
|
if (f != NULL)
|
|
{
|
|
fgets (line, 1024, f);
|
|
while (!feof (f))
|
|
{
|
|
channel = 0;
|
|
if (sscanf (line, "CMD%1d", &channel) != 1)
|
|
channel = 0;
|
|
switch (channel)
|
|
{
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
base = opsc04;
|
|
max = 8;
|
|
break;
|
|
case 2:
|
|
base = opsc53;
|
|
max = 16;
|
|
break;
|
|
case 8:
|
|
base = opscan;
|
|
max = 36;
|
|
break;
|
|
default:
|
|
channel = 0;
|
|
}
|
|
printf ("CMD%d BEFORE: ", channel);
|
|
for (i = 0; i < max; i++)
|
|
printf ("%02X ", base[i]);
|
|
printf ("\n");
|
|
if (channel > 0)
|
|
{
|
|
ptr = line + 6;
|
|
for (i = 0; (i < max) && ((ptr - line) < strlen (line)); i++)
|
|
{
|
|
if (ptr[0] != '-')
|
|
{
|
|
sscanf (ptr, "%X", base + i);
|
|
}
|
|
ptr += 3;
|
|
}
|
|
}
|
|
printf ("CMD%d AFTER : ", channel);
|
|
for (i = 0; i < max; i++)
|
|
printf ("%02X ", base[i]);
|
|
printf ("\n");
|
|
fgets (line, 1024, f);
|
|
}
|
|
fclose (f);
|
|
}
|
|
#endif
|
|
|
|
|
|
CMDSETGET (2, 0x10, opsc53);
|
|
CMDSETGET (8, 0x24, opscan);
|
|
CMDSETGET (1, 0x08, opsc04);
|
|
CMDSYNC (0xC2);
|
|
CMDSET (4, 0x3EC6, calibration);
|
|
COMPLETIONWAIT;
|
|
|
|
|
|
*rbpp = bpp;
|
|
*rtw = tw;
|
|
*rth = th;
|
|
|
|
free (buffer);
|
|
return (1);
|
|
}
|
|
|
|
/*
|
|
* Check the scanner model. Return 1220 for
|
|
* a 1220P, or 2000 for a 2000P.
|
|
* and 610 for a 610P
|
|
* values less than 610 are errors
|
|
*/
|
|
int
|
|
sanei_umax_pp_CheckModel (void)
|
|
{
|
|
int *dest = NULL;
|
|
int state[16];
|
|
int err = 0;
|
|
int i;
|
|
|
|
int opsc35[37] =
|
|
{ 0x00, 0x00, 0x04, 0x00, 0x02, 0x00, 0x00, 0x0C, 0x00, 0x03, 0xC1, 0x80,
|
|
0x00, 0x00, 0x04, 0x00, 0x16, 0x41, 0xE0, 0xAC, 0x03, 0x03, 0x00, 0x00,
|
|
0x46, 0xA0, 0x00, 0x8B, 0x49, 0x2A, 0xE9, 0x68, 0xDF, 0x13, 0x1A, 0x00,
|
|
-1
|
|
};
|
|
|
|
/* if we have already detected a scanner different from */
|
|
/* default type, no need to check again */
|
|
if (sanei_umax_pp_getastra ())
|
|
return sanei_umax_pp_getastra ();
|
|
|
|
/* get scanner status */
|
|
CMDGET (0x02, 16, state);
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
|
|
dest = (int *) malloc (65536 * 4);
|
|
if (dest == NULL)
|
|
{
|
|
DBG (0, "%s:%d failed to allocate 256 Ko !\n", __FILE__, __LINE__);
|
|
return (0);
|
|
}
|
|
|
|
/* init some buffer : default calibration data ? */
|
|
dest[0] = 0x00;
|
|
dest[1] = 0x00;
|
|
dest[2] = 0x00;
|
|
for (i = 0; i < 768; i++)
|
|
dest[i + 3] = i % 256;
|
|
dest[768 + 3] = 0xAA;
|
|
dest[768 + 4] = 0xAA;
|
|
dest[768 + 5] = -1;
|
|
CMDSETGET (0x04, 768 + 5, dest);
|
|
|
|
|
|
/* check buffer returned */
|
|
for (i = 0; i < 768; i++)
|
|
{
|
|
if (dest[i + 3] != (i % 256))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Error data altered: byte %d=0x%02X, should be 0x%02X ! (%s:%d)\n",
|
|
i, dest[i + 3], i % 256, __FILE__, __LINE__);
|
|
err = 1;
|
|
}
|
|
}
|
|
if (err)
|
|
return (0);
|
|
|
|
|
|
/* new part of buffer ... */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
dest[i * 2] = i;
|
|
dest[i * 2 + 1] = 0x00;
|
|
}
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
CMDSET (0x04, 512, dest);
|
|
|
|
/* another new part ... */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
dest[i * 2] = i;
|
|
dest[i * 2 + 1] = 0x04; /* instead of 0x00 */
|
|
}
|
|
opsc35[2] = 0x06; /* instead of 0x04, write flag ? */
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDSYNC (0xC2);
|
|
CMDSET (0x04, 512, dest);
|
|
|
|
opsc35[2] = 0x04; /* return to initial value, read flag? */
|
|
CMDSETGET (0x08, 36, opsc35);
|
|
CMDGET (0x04, 512, dest);
|
|
|
|
/* check buffer returned */
|
|
/* if 0x4 are still 0x04, we got a 1220P, else it is a 2000P */
|
|
for (i = 0; i < 256; i++)
|
|
{
|
|
if ((dest[2 * i] != i)
|
|
|| ((dest[2 * i + 1] != 0x04) && (dest[2 * i + 1] != 0x00)))
|
|
{
|
|
DBG
|
|
(0,
|
|
"Error data altered: expected %d=(0x%02X,0x04), found (0x%02X,0x%02X) ! (%s:%d)\n",
|
|
i, i, dest[i * 2], dest[i * 2 + 1], __FILE__, __LINE__);
|
|
err = 0;
|
|
}
|
|
}
|
|
|
|
/* if buffer unchanged, we have a 1600P, or a 1220P */
|
|
/* if data has turned into 0, we have a 2000P */
|
|
if (dest[1] == 0x00)
|
|
{
|
|
sanei_umax_pp_setastra (2000);
|
|
err = 2000;
|
|
}
|
|
else
|
|
{
|
|
/* detects 1600 by finding black scans */
|
|
/* we defaults to 1220 */
|
|
sanei_umax_pp_setastra (1220);
|
|
MoveToOrigin ();
|
|
err = sanei_umax_pp_getastra ();
|
|
|
|
/* parking */
|
|
CMDSYNC (0xC2);
|
|
CMDSYNC (0x00);
|
|
if (sanei_umax_pp_Park () == 0)
|
|
DBG (0, "Park failed !!! (%s:%d)\n", __FILE__, __LINE__);
|
|
|
|
/* poll parking */
|
|
do
|
|
{
|
|
sleep (1);
|
|
CMDSYNC (0x40);
|
|
}
|
|
while ((sanei_umax_pp_ScannerStatus () & MOTOR_BIT) == 0x00);
|
|
}
|
|
|
|
/* return guessed model number */
|
|
CMDSYNC (0x00);
|
|
return (err);
|
|
}
|
|
|
|
|
|
|
|
/* sets, resets gamma tables */
|
|
|
|
void
|
|
sanei_umax_pp_gamma (int *red, int *green, int *blue)
|
|
{
|
|
if (red != NULL)
|
|
{
|
|
ggRed = red;
|
|
}
|
|
else
|
|
{
|
|
ggRed = ggamma;
|
|
}
|
|
|
|
if (green != NULL)
|
|
{
|
|
ggGreen = green;
|
|
}
|
|
else
|
|
{
|
|
ggGreen = ggamma;
|
|
}
|
|
|
|
if (blue != NULL)
|
|
{
|
|
ggBlue = blue;
|
|
}
|
|
else
|
|
{
|
|
ggBlue = ggamma;
|
|
}
|
|
}
|