/*............................................................................. * Project : SANE library for Plustek flatbed scanners. *............................................................................. */ /** @file plustek-usbimg.c * @brief Image processing functions for copying and scaling image lines. * * Based on sources acquired from Plustek Inc.<br> * Copyright (C) 2001-2007 Gerhard Jaeger <gerhard@gjaeger.de> * * History: * - 0.40 - starting version of the USB support * - 0.41 - fixed the 14bit problem for LM9831 devices * - 0.42 - no changes * - 0.43 - no changes * - 0.44 - added CIS parts and dumpPic function * - 0.45 - added gray scaling functions for CIS devices * - fixed usb_GrayScale16 function * - fixed a bug in usb_ColorScale16_2 function * - fixed endless loop bug * - fixed a bug in usb_GrayScalePseudo16 function * - fixed a bug in usb_GrayDuplicatePseudo16 function * - removed the scaler stuff for CIS devices * - 0.46 - minor fixes * - 0.47 - added big-endian/little endian stuff * - 0.48 - fixed usb_ColorDuplicateGray16() and * usb_ColorScaleGray16() * - added usb_BWScaleFromColor() and usb_BWDuplicateFromColor() * - cleanup * - 0.49 - a_bRegs is now part of the device structure * - 0.50 - cleanup * - 0.51 - added usb_ColorDuplicateGray16_2(), usb_ColorScaleGray16_2() * usb_BWScaleFromColor_2() and usb_BWDuplicateFromColor_2() * - 0.52 - cleanup * . * <hr> * This file is part of the SANE package. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, * MA 02111-1307, USA. * * As a special exception, the authors of SANE give permission for * additional uses of the libraries contained in this release of SANE. * * The exception is that, if you link a SANE library with other files * to produce an executable, this does not by itself cause the * resulting executable to be covered by the GNU General Public * License. Your use of that executable is in no way restricted on * account of linking the SANE library code into it. * * This exception does not, however, invalidate any other reasons why * the executable file might be covered by the GNU General Public * License. * * If you submit changes to SANE to the maintainers to be included in * a subsequent release, you agree by submitting the changes that * those changes may be distributed with this exception intact. * * If you write modifications of your own for SANE, it is your choice * whether to permit this exception to apply to your modifications. * If you do not wish that, delete this exception notice. * <hr> */ #define _SCALER 1000 static u_char bShift, Shift; static u_short wSum, Mask; /* */ static u_char BitTable[8] = { 0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01 }; static u_char BitsReverseTable[256] = { 0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0, 0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0, 0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8, 0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8, 0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4, 0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4, 0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec, 0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc, 0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2, 0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2, 0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea, 0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa, 0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6, 0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6, 0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee, 0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe, 0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1, 0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1, 0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9, 0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9, 0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5, 0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5, 0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed, 0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd, 0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3, 0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3, 0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb, 0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb, 0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7, 0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7, 0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef, 0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff }; /************************ some helper functions ******************************/ static void ReverseBits( int b, u_char **pTar, int *iByte, int *iWeightSum, int iSrcWeight, int iTarWeight, int cMax ) { int bit; cMax = (1 << cMax); if(iSrcWeight == iTarWeight) { for( bit = 1; bit < cMax; bit <<= 1) { *iByte <<= 1; if(b & bit) *iByte |= 1; if(*iByte >= 0x100) { **pTar++ = (u_char)*iByte; *iByte = 1; } } } else { for( bit = 1; bit < cMax; bit <<= 1 ) { *iWeightSum += iTarWeight; while(*iWeightSum >= iSrcWeight) { *iWeightSum -= iSrcWeight; *iByte <<= 1; if(b & bit) *iByte |= 1; if(*iByte >= 0x100) { **pTar++ = (u_char)*iByte; *iByte = 1; } } } } } static void usb_ReverseBitStream( u_char *pSrc, u_char *pTar, int iPixels, int iBufSize, int iSrcWeight/* = 0*/, int iTarWeight/* = 0*/, int iPadBit/* = 1*/) { int i; int iByte = 1; int cBytes = iPixels / 8; int cBits = iPixels % 8; u_char bPad = iPadBit? 0xff: 0; u_char *pTarget = pTar; int iWeightSum = 0; if(iSrcWeight == iTarWeight) { if(cBits) { int cShift = 8 - cBits; for(i = cBytes, pSrc = pSrc + cBytes - 1; i > 0; i--, pSrc--, pTar++) *pTar = BitsReverseTable[(u_char)((*pSrc << cBits) | (*(pSrc+1) >> cShift))]; ReverseBits(*(pSrc+1) >> cShift, &pTar, &iByte, &iWeightSum, iSrcWeight, iTarWeight, cBits); } else /* byte boundary */ { for(i = cBytes, pSrc = pSrc + cBytes - 1; i > 0; i--, pSrc--, pTar++) *pTar = BitsReverseTable[*pSrc]; } } else /* To shrink or enlarge */ { if(cBits) { int cShift = 8 - cBits; for(i = cBytes, pSrc = pSrc + cBytes - 1; i > 0; i--, pSrc--) ReverseBits((*pSrc << cBits) | (*(pSrc+1) >> cShift), &pTar, &iByte, &iWeightSum, iSrcWeight, iTarWeight, 8); ReverseBits(*(pSrc+1) >> cShift, &pTar, &iByte, &iWeightSum, iSrcWeight, iTarWeight, cBits); } else /* byte boundary */ { for(i = cBytes, pSrc = pSrc + cBytes - 1; i > 0; i--, pSrc--) ReverseBits(*pSrc, &pTar, &iByte, &iWeightSum, iSrcWeight, iTarWeight, 8); } } if(iByte != 1) { while(iByte < 0x100) { iByte <<= 1; iByte |= iPadBit; } *pTar++ = (u_char)iByte; } cBytes = (int)(pTar - pTarget); for(i = iBufSize - cBytes; i > 0; i--, pTar++) *pTar = bPad; } /** */ static void usb_AverageColorByte( Plustek_Device *dev ) { u_long dw; ScanDef *scan = &dev->scanning; if((scan->sParam.bSource == SOURCE_Negative || scan->sParam.bSource == SOURCE_Transparency) && scan->sParam.PhyDpi.x > 800) { for (dw = 0; dw < (scan->sParam.Size.dwPhyPixels - 1); dw++) { scan->Red.pcb[dw].a_bColor[0] = (u_char)(((u_short)scan->Red.pcb[dw].a_bColor[0] + (u_short)scan->Red.pcb[dw + 1].a_bColor[0]) / 2); scan->Green.pcb[dw].a_bColor[0] = (u_char)(((u_short)scan->Green.pcb[dw].a_bColor[0] + (u_short)scan->Green.pcb[dw + 1].a_bColor[0]) / 2); scan->Blue.pcb[dw].a_bColor[0] = (u_char)(((u_short)scan->Blue.pcb[dw].a_bColor[0] + (u_short)scan->Blue.pcb[dw + 1].a_bColor[0]) / 2); } } } /** */ static void usb_AverageColorWord( Plustek_Device *dev ) { u_char ls = 2; u_long dw; ScanDef *scan = &dev->scanning; if((scan->sParam.bSource == SOURCE_Negative || scan->sParam.bSource == SOURCE_Transparency) && scan->sParam.PhyDpi.x > 800) { scan->Red.pcw[0].Colors[0] = _HILO2WORD(scan->Red.pcw[0].HiLo[0]) >> ls; scan->Green.pcw[0].Colors[0] = _HILO2WORD(scan->Green.pcw[0].HiLo[0]) >> ls; scan->Blue.pcw[0].Colors[0] = _HILO2WORD(scan->Blue.pcw[0].HiLo[0]) >> ls; for (dw = 0; dw < (scan->sParam.Size.dwPhyPixels - 1); dw++) { scan->Red.pcw[dw + 1].Colors[0] = _HILO2WORD(scan->Red.pcw[dw + 1].HiLo[0]) >> ls; scan->Green.pcw[dw + 1].Colors[0] = _HILO2WORD(scan->Green.pcw[dw + 1].HiLo[0]) >> ls; scan->Blue.pcw[dw + 1].Colors[0] = _HILO2WORD(scan->Blue.pcw[dw + 1].HiLo[0]) >> ls; scan->Red.pcw[dw].Colors[0] = (u_short)(((u_long)scan->Red.pcw[dw].Colors[0] + (u_long)scan->Red.pcw[dw + 1].Colors[0]) / 2); scan->Green.pcw[dw].Colors[0] = (u_short)(((u_long)scan->Green.pcw[dw].Colors[0] + (u_long)scan->Green.pcw[dw + 1].Colors[0]) / 2); scan->Blue.pcw[dw].Colors[0] = (u_short)(((u_long)scan->Blue.pcw[dw].Colors[0] + (u_long)scan->Blue.pcw[dw + 1].Colors[0]) / 2); scan->Red.pcw[dw].Colors[0] = _HILO2WORD(scan->Red.pcw[dw].HiLo[0]) << ls; scan->Green.pcw[dw].Colors[0] = _HILO2WORD(scan->Green.pcw[dw].HiLo[0]) << ls; scan->Blue.pcw[dw].Colors[0] = _HILO2WORD(scan->Blue.pcw[dw].HiLo[0]) << ls; } scan->Red.pcw[dw].Colors[0] = _HILO2WORD(scan->Red.pcw[dw].HiLo[0]) << ls; scan->Green.pcw[dw].Colors[0] = _HILO2WORD(scan->Green.pcw[dw].HiLo[0]) << ls; scan->Blue.pcw[dw].Colors[0] = _HILO2WORD(scan->Blue.pcw[dw].HiLo[0]) << ls; } } /** */ static void usb_AverageGrayByte( Plustek_Device *dev ) { u_long dw; ScanDef *scan = &dev->scanning; if((scan->sParam.bSource == SOURCE_Negative || scan->sParam.bSource == SOURCE_Transparency) && scan->sParam.PhyDpi.x > 800) { for (dw = 0; dw < (scan->sParam.Size.dwPhyPixels - 1); dw++) scan->Green.pb[dw] = (u_char)(((u_short)scan->Green.pb[dw]+ (u_short)scan->Green.pb[dw+1]) / 2); } } /** */ static void usb_AverageGrayWord( Plustek_Device *dev ) { u_long dw; ScanDef *scan = &dev->scanning; if((scan->sParam.bSource == SOURCE_Negative || scan->sParam.bSource == SOURCE_Transparency) && scan->sParam.PhyDpi.x > 800) { scan->Green.pw[0] = _HILO2WORD(scan->Green.philo[0]) >> 2; for (dw = 0; dw < (scan->sParam.Size.dwPhyPixels - 1); dw++) { scan->Green.pw[dw + 1] = _HILO2WORD(scan->Green.philo[dw+1]) >> 2; scan->Green.pw[dw] = (u_short)(((u_long)scan->Green.pw[dw]+ (u_long)scan->Green.pw[dw+1]) / 2); scan->Green.pw[dw] = _HILO2WORD(scan->Green.philo[dw]) << 2; } scan->Green.pw[dw] = _HILO2WORD(scan->Green.philo[dw]) << 2; } } /** * returns the zoom value, used for our scaling algorithm (DDA algo * digital differential analyzer). */ static int usb_GetScaler( ScanDef *scan ) { double ratio; ratio = (double)scan->sParam.UserDpi.x/ (double)scan->sParam.PhyDpi.x; return (int)(1.0/ratio * _SCALER); } /******************************* the copy functions **************************/ /** do a simple memcopy from scan-buffer to user buffer */ static void usb_ColorDuplicate8( Plustek_Device *dev ) { int next; u_long dw, pixels; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } for( dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next ) { scan->UserBuf.pb_rgb[pixels].Red = scan->Red.pcb[dw].a_bColor[0]; scan->UserBuf.pb_rgb[pixels].Green = scan->Green.pcb[dw].a_bColor[0]; scan->UserBuf.pb_rgb[pixels].Blue = scan->Blue.pcb[dw].a_bColor[0]; } } /** reorder from rgb line to rgb pixel (CIS scanner) */ static void usb_ColorDuplicate8_2( Plustek_Device *dev ) { int next; u_long dw, pixels; ScanDef *scan = &dev->scanning; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } for( dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next ) { scan->UserBuf.pb_rgb[pixels].Red = (u_char)scan->Red.pb[dw]; scan->UserBuf.pb_rgb[pixels].Green = (u_char)scan->Green.pb[dw]; scan->UserBuf.pb_rgb[pixels].Blue = (u_char)scan->Blue.pb[dw]; } } /** */ static void usb_ColorDuplicate16( Plustek_Device *dev ) { int next; u_char ls; u_long dw, pixels; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageColorWord( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { if( swap ) { scan->UserBuf.pw_rgb[pixels].Red = _HILO2WORD(scan->Red.pcw[dw].HiLo[0]) >> ls; scan->UserBuf.pw_rgb[pixels].Green = _HILO2WORD(scan->Green.pcw[dw].HiLo[0]) >> ls; scan->UserBuf.pw_rgb[pixels].Blue = _HILO2WORD(scan->Blue.pcw[dw].HiLo[0]) >> ls; } else { scan->UserBuf.pw_rgb[pixels].Red = scan->Red.pw[dw] >> ls; scan->UserBuf.pw_rgb[pixels].Green= scan->Green.pw[dw] >> ls; scan->UserBuf.pw_rgb[pixels].Blue = scan->Blue.pw[dw] >> ls; } } } /** */ static void usb_ColorDuplicate16_2( Plustek_Device *dev ) { int next; u_char ls; HiLoDef tmp; u_long dw, pixels; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageColorWord( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; for( dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { if( swap ) { tmp = *((HiLoDef*)&scan->Red.pw[dw]); scan->UserBuf.pw_rgb[pixels].Red = _HILO2WORD(tmp) >> ls; tmp = *((HiLoDef*)&scan->Green.pw[dw]); scan->UserBuf.pw_rgb[pixels].Green = _HILO2WORD(tmp) >> ls; tmp = *((HiLoDef*)&scan->Blue.pw[dw]); scan->UserBuf.pw_rgb[pixels].Blue = _HILO2WORD(tmp) >> ls; } else { scan->UserBuf.pw_rgb[pixels].Red = scan->Red.pw[dw] >> ls; scan->UserBuf.pw_rgb[pixels].Green = scan->Green.pw[dw] >> ls; scan->UserBuf.pw_rgb[pixels].Blue = scan->Blue.pw[dw] >> ls; } } } /** */ static void usb_ColorDuplicatePseudo16( Plustek_Device *dev ) { int next; u_short wR, wG, wB; u_long dw, pixels; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } wR = (u_short)scan->Red.pcb[0].a_bColor[0]; wG = (u_short)scan->Green.pcb[0].a_bColor[0]; wB = (u_short)scan->Blue.pcb[0].a_bColor[0]; for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { scan->UserBuf.pw_rgb[pixels].Red = (wR + scan->Red.pcb[dw].a_bColor[0]) << bShift; scan->UserBuf.pw_rgb[pixels].Green = (wG + scan->Green.pcb[dw].a_bColor[0]) << bShift; scan->UserBuf.pw_rgb[pixels].Blue = (wB + scan->Blue.pcb[dw].a_bColor[0]) << bShift; wR = (u_short)scan->Red.pcb[dw].a_bColor[0]; wG = (u_short)scan->Green.pcb[dw].a_bColor[0]; wB = (u_short)scan->Blue.pcb[dw].a_bColor[0]; } } /** */ static void usb_ColorDuplicateGray( Plustek_Device *dev ) { int next; u_long dw, pixels; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } switch(scan->fGrayFromColor) { case 1: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Red.pcb[dw].a_bColor[0]; break; case 2: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Green.pcb[dw].a_bColor[0]; break; case 3: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Blue.pcb[dw].a_bColor[0]; break; } } /** */ static void usb_ColorDuplicateGray_2( Plustek_Device *dev ) { int next; u_long dw, pixels; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } switch(scan->fGrayFromColor) { case 1: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Red.pb[dw]; break; case 3: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Blue.pb[dw]; break; default: for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pb[pixels] = scan->Green.pb[dw]; break; } } /** */ static void usb_ColorDuplicateGray16( Plustek_Device *dev ) { int next; u_char ls; u_long dw, pixels; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageColorWord( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; switch(scan->fGrayFromColor) { case 1: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Red.pcw[dw].HiLo[0]) >> ls; } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = scan->Red.pw[dw] >> ls; } break; case 2: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Green.pcw[dw].HiLo[0]) >> ls; } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = scan->Green.pw[dw] >> ls; } break; case 3: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Blue.pcw[dw].HiLo[0]) >> ls; } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) scan->UserBuf.pw[pixels] = scan->Blue.pw[dw] >> ls; } break; } } /** */ static void usb_ColorDuplicateGray16_2( Plustek_Device *dev ) { int next; u_char ls; u_long dw, pixels; HiLoDef tmp; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageColorWord( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; switch(scan->fGrayFromColor) { case 1: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { tmp = *((HiLoDef*)&scan->Red.pw[dw]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { scan->UserBuf.pw[pixels] = scan->Red.pw[dw] >> ls; } } break; case 2: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { tmp = *((HiLoDef*)&scan->Green.pw[dw]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { scan->UserBuf.pw[pixels] = scan->Green.pw[dw] >> ls; } } break; case 3: if( swap ) { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { tmp = *((HiLoDef*)&scan->Blue.pw[dw]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } } else { for (dw = 0; dw < scan->sParam.Size.dwPixels; dw++, pixels += next) { scan->UserBuf.pw[pixels] = scan->Blue.pw[dw] >> ls; } } break; } } /** */ static void usb_GrayDuplicate8( Plustek_Device *dev ) { u_char *dest, *src; u_long pixels; ScanDef *scan = &dev->scanning; usb_AverageGrayByte( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { pixels = scan->sParam.Size.dwPixels; src = scan->Green.pb; dest = scan->UserBuf.pb + pixels - 1; for(; pixels; pixels--, src++, dest--) *dest = *src; } else { memcpy( scan->UserBuf.pb, scan->Green.pb, scan->sParam.Size.dwBytes ); } } /** */ static void usb_GrayDuplicate16( Plustek_Device *dev ) { int next; u_char ls; u_short *dest; u_long pixels; HiLoDef *pwm; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageGrayWord( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; dest = scan->UserBuf.pw + scan->sParam.Size.dwPixels - 1; } else { next = 1; dest = scan->UserBuf.pw; } if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; pwm = scan->Green.philo; for( pixels=scan->sParam.Size.dwPixels; pixels--; pwm++, dest += next ) { if( swap ) *dest = (_PHILO2WORD(pwm)) >> ls; else *dest = (_PLOHI2WORD(pwm)) >> ls; } } /** */ static void usb_GrayDuplicatePseudo16( Plustek_Device *dev ) { u_char *src; int next; u_short g; u_short *dest; u_long pixels; ScanDef *scan = &dev->scanning; usb_AverageGrayByte( dev ); if (scan->sParam.bSource == SOURCE_ADF) { next = -1; dest = scan->UserBuf.pw + scan->sParam.Size.dwPixels - 1; } else { next = 1; dest = scan->UserBuf.pw; } src = scan->Green.pb; g = (u_short)*src; for( pixels=scan->sParam.Size.dwPixels; pixels--; src++, dest += next ) { *dest = (g + *src) << bShift; g = (u_short)*src; } } /** copy binary data to the user buffer */ static void usb_BWDuplicate( Plustek_Device *dev ) { ScanDef *scan = &dev->scanning; if(scan->sParam.bSource == SOURCE_ADF) { usb_ReverseBitStream( scan->Green.pb, scan->UserBuf.pb, scan->sParam.Size.dwValidPixels, scan->dwBytesLine, 0, 0, 1 ); } else { memcpy( scan->UserBuf.pb, scan->Green.pb, scan->sParam.Size.dwBytes ); } } /** generate binary data from one of the three color inputs according to the * value in fGrayFromColor (CCD version) */ static void usb_BWDuplicateFromColor( Plustek_Device *dev ) { int next; u_char d, s, *dest; u_short j; u_long pixels; ColorByteDef *src; ScanDef *scan = &dev->scanning; if( scan->sParam.bSource == SOURCE_ADF ) { dest = scan->UserBuf.pb + scan->sParam.Size.dwPixels - 1; next = -1; } else { dest = scan->UserBuf.pb; next = 1; } switch(scan->fGrayFromColor) { case 1: src = scan->Red.pcb; break; case 3: src = scan->Blue.pcb; break; default: src = scan->Green.pcb; break; } d = j = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; pixels--, src++ ) { s = src->a_bColor[0]; if( s != 0 ) d |= BitTable[j]; j++; if( j == 8 ) { *dest = d; dest += next; d = j = 0; } } } /** generate binary data from one of the three color inputs according to the * value in fGrayFromColor (CIS version) */ static void usb_BWDuplicateFromColor_2( Plustek_Device *dev ) { int next; u_char d, *dest, *src; u_short j; u_long pixels; ScanDef *scan = &dev->scanning; if( scan->sParam.bSource == SOURCE_ADF ) { dest = scan->UserBuf.pb + scan->sParam.Size.dwPixels - 1; next = -1; } else { dest = scan->UserBuf.pb; next = 1; } switch(scan->fGrayFromColor) { case 1: src = scan->Red.pb; break; case 3: src = scan->Blue.pb; break; default: src = scan->Green.pb; break; } d = j = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; pixels--, src++ ) { if( *src != 0 ) d |= BitTable[j]; j++; if( j == 8 ) { *dest = d; dest += next; d = j = 0; } } } /************************** the scaling functions ****************************/ /** */ static void usb_ColorScaleGray( Plustek_Device *dev ) { int izoom, ddax, next; u_long dw, pixels; ColorByteDef *src; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } switch(scan->fGrayFromColor) { case 1: src = scan->Red.pcb; break; case 3: src = scan->Blue.pcb; break; default: src = scan->Green.pcb; break; } izoom = usb_GetScaler( scan ); for( ddax = 0; dw; src++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { scan->UserBuf.pb[pixels] = src->a_bColor[0]; pixels += next; ddax += izoom; dw--; } } } /** */ static void usb_ColorScaleGray_2( Plustek_Device *dev ) { u_char *src; int izoom, ddax, next; u_long dw, pixels; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } switch(scan->fGrayFromColor) { case 1: src = scan->Red.pb; break; case 3: src = scan->Blue.pb; break; default: src = scan->Green.pb; break; } izoom = usb_GetScaler( scan ); for( ddax = 0; dw; src++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { scan->UserBuf.pb[pixels] = *src; pixels += next; ddax += izoom; dw--; } } } /** */ static void usb_ColorScaleGray16( Plustek_Device *dev ) { u_char ls; int izoom, ddax, next; u_long dw, pixels, bitsput; SANE_Bool swap = usb_HostSwap(); ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; switch( scan->fGrayFromColor ) { case 1: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Red.pcw[bitsput].HiLo[0]) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Red.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } break; case 2: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Green.pcw[bitsput].HiLo[0]) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Green.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } break; case 3: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { scan->UserBuf.pw[pixels] = _HILO2WORD(scan->Blue.pcw[bitsput].HiLo[0]) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Blue.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } break; } } /** */ static void usb_ColorScaleGray16_2( Plustek_Device *dev ) { u_char ls; int izoom, ddax, next; u_long dw, pixels, bitsput; HiLoDef tmp; SANE_Bool swap = usb_HostSwap(); ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; switch( scan->fGrayFromColor ) { case 1: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { tmp = *((HiLoDef*)&scan->Red.pw[bitsput]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Red.pw[dw] >> ls; } pixels += next; ddax += izoom; dw--; } } break; case 2: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { tmp = *((HiLoDef*)&scan->Green.pw[bitsput]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Green.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } break; case 3: for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { tmp = *((HiLoDef*)&scan->Blue.pw[bitsput]); scan->UserBuf.pw[pixels] = _HILO2WORD(tmp) >> ls; } else { scan->UserBuf.pw[pixels] = scan->Blue.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } break; } } /** here we copy and scale from scanner world to user world... */ static void usb_ColorScale8( Plustek_Device *dev ) { int izoom, ddax, next; u_long dw, pixels, bitsput; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { scan->UserBuf.pb_rgb[pixels].Red = scan->Red.pcb[bitsput].a_bColor[0]; scan->UserBuf.pb_rgb[pixels].Green = scan->Green.pcb[bitsput].a_bColor[0]; scan->UserBuf.pb_rgb[pixels].Blue = scan->Blue.pcb[bitsput].a_bColor[0]; pixels += next; ddax += izoom; dw--; } } } static void usb_ColorScale8_2( Plustek_Device *dev ) { int izoom, ddax, next; u_long dw, pixels, bitsput; ScanDef *scan = &dev->scanning; dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { scan->UserBuf.pb_rgb[pixels].Red = scan->Red.pb[bitsput]; scan->UserBuf.pb_rgb[pixels].Green = scan->Green.pb[bitsput]; scan->UserBuf.pb_rgb[pixels].Blue = scan->Blue.pb[bitsput]; pixels += next; ddax += izoom; dw--; } } } /** */ static void usb_ColorScale16( Plustek_Device *dev ) { u_char ls; int izoom, ddax, next; u_long dw, pixels, bitsput; SANE_Bool swap = usb_HostSwap(); ScanDef *scan = &dev->scanning; usb_AverageColorWord( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { scan->UserBuf.pw_rgb[pixels].Red = _HILO2WORD(scan->Red.pcw[bitsput].HiLo[0]) >> ls; scan->UserBuf.pw_rgb[pixels].Green = _HILO2WORD(scan->Green.pcw[bitsput].HiLo[0]) >> ls; scan->UserBuf.pw_rgb[pixels].Blue = _HILO2WORD(scan->Blue.pcw[bitsput].HiLo[0]) >> ls; } else { scan->UserBuf.pw_rgb[pixels].Red = scan->Red.pw[bitsput]>>ls; scan->UserBuf.pw_rgb[pixels].Green = scan->Green.pw[bitsput] >> ls; scan->UserBuf.pw_rgb[pixels].Blue = scan->Blue.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } } /** */ static void usb_ColorScale16_2( Plustek_Device *dev ) { u_char ls; HiLoDef tmp; int izoom, ddax, next; u_long dw, pixels, bitsput; SANE_Bool swap = usb_HostSwap(); ScanDef *scan = &dev->scanning; usb_AverageColorWord( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { if( swap ) { tmp = *((HiLoDef*)&scan->Red.pw[bitsput]); scan->UserBuf.pw_rgb[pixels].Red = _HILO2WORD(tmp) >> ls; tmp = *((HiLoDef*)&scan->Green.pw[bitsput]); scan->UserBuf.pw_rgb[pixels].Green = _HILO2WORD(tmp) >> ls; tmp = *((HiLoDef*)&scan->Blue.pw[bitsput]); scan->UserBuf.pw_rgb[pixels].Blue = _HILO2WORD(tmp) >> ls; } else { scan->UserBuf.pw_rgb[pixels].Red = scan->Red.pw[bitsput] >> ls; scan->UserBuf.pw_rgb[pixels].Green = scan->Green.pw[bitsput] >> ls; scan->UserBuf.pw_rgb[pixels].Blue = scan->Blue.pw[bitsput] >> ls; } pixels += next; ddax += izoom; dw--; } } } /** */ static void usb_ColorScalePseudo16( Plustek_Device *dev ) { int izoom, ddax, next; u_short wR, wG, wB; u_long dw, pixels, bitsput; ScanDef *scan = &dev->scanning; usb_AverageColorByte( dev ); dw = scan->sParam.Size.dwPixels; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; pixels = scan->sParam.Size.dwPixels - 1; } else { next = 1; pixels = 0; } izoom = usb_GetScaler( scan ); wR = (u_short)scan->Red.pcb[0].a_bColor[0]; wG = (u_short)scan->Green.pcb[0].a_bColor[1]; wB = (u_short)scan->Blue.pcb[0].a_bColor[2]; for( bitsput = 0, ddax = 0; dw; bitsput++ ) { ddax -= _SCALER; while((ddax < 0) && (dw > 0)) { scan->UserBuf.pw_rgb[pixels].Red = (wR + scan->Red.pcb[bitsput].a_bColor[0]) << bShift; scan->UserBuf.pw_rgb[pixels].Green = (wG + scan->Green.pcb[bitsput].a_bColor[0]) << bShift; scan->UserBuf.pw_rgb[pixels].Blue = (wB + scan->Blue.pcb[bitsput].a_bColor[0]) << bShift; pixels += next; ddax += izoom; dw--; } wR = (u_short)scan->Red.pcb[bitsput].a_bColor[0]; wG = (u_short)scan->Green.pcb[bitsput].a_bColor[0]; wB = (u_short)scan->Blue.pcb[bitsput].a_bColor[0]; } } /** */ static void usb_BWScale( Plustek_Device *dev ) { u_char tmp, *dest, *src; int izoom, ddax; u_long i, dw; ScanDef *scan = &dev->scanning; src = scan->Green.pb; if( scan->sParam.bSource == SOURCE_ADF ) { int iSum = wSum; usb_ReverseBitStream(scan->Green.pb, scan->UserBuf.pb, scan->sParam.Size.dwValidPixels, scan->dwBytesLine, scan->sParam.PhyDpi.x, scan->sParam.UserDpi.x, 1 ); wSum = iSum; return; } else { dest = scan->UserBuf.pb; } izoom = usb_GetScaler( scan ); memset( dest, 0, scan->dwBytesLine ); ddax = 0; dw = 0; for( i = 0; i < scan->sParam.Size.dwValidPixels; i++ ) { ddax -= _SCALER; while( ddax < 0 ) { tmp = src[(i>>3)]; if((dw>>3) < scan->sParam.Size.dwValidPixels ) { if( 0 != (tmp &= (1 << ((~(i & 0x7))&0x7)))) dest[dw>>3] |= (1 << ((~(dw & 0x7))&0x7)); } dw++; ddax += izoom; } } } /** */ static void usb_BWScaleFromColor( Plustek_Device *dev ) { u_char d, s, *dest; u_short j; u_long pixels; int izoom, ddax, next; ColorByteDef *src; ScanDef *scan = &dev->scanning; if (scan->sParam.bSource == SOURCE_ADF) { dest = scan->UserBuf.pb + scan->sParam.Size.dwPixels - 1; next = -1; } else { dest = scan->UserBuf.pb; next = 1; } /* setup the source buffer */ switch(scan->fGrayFromColor) { case 1: src = scan->Red.pcb; break; case 3: src = scan->Blue.pcb; break; default: src = scan->Green.pcb; break; } izoom = usb_GetScaler( scan ); ddax = 0; d = j = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; src++ ) { ddax -= _SCALER; while((ddax < 0) && (pixels > 0)) { s = src->a_bColor[0]; if( s != 0 ) d |= BitTable[j]; j++; if( j == 8 ) { *dest = d; dest += next; d = j = 0; } ddax += izoom; pixels--; } } } /** */ static void usb_BWScaleFromColor_2( Plustek_Device *dev ) { u_char d, *dest, *src; u_short j; u_long pixels; int izoom, ddax, next; ScanDef *scan = &dev->scanning; if (scan->sParam.bSource == SOURCE_ADF) { dest = scan->UserBuf.pb + scan->sParam.Size.dwPixels - 1; next = -1; } else { dest = scan->UserBuf.pb; next = 1; } /* setup the source buffer */ switch(scan->fGrayFromColor) { case 1: src = scan->Red.pb; break; case 3: src = scan->Blue.pb; break; default: src = scan->Green.pb; break; } izoom = usb_GetScaler( scan ); ddax = 0; d = j = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; src++ ) { ddax -= _SCALER; while((ddax < 0) && (pixels > 0)) { if( *src != 0 ) d |= BitTable[j]; j++; if( j == 8 ) { *dest = d; dest += next; d = j = 0; } ddax += izoom; pixels--; } } } /** */ static void usb_GrayScale8( Plustek_Device *dev ) { u_char *dest, *src; int izoom, ddax, next; u_long pixels; ScanDef *scan = &dev->scanning; usb_AverageGrayByte( dev ); src = scan->Green.pb; if( scan->sParam.bSource == SOURCE_ADF ) { dest = scan->UserBuf.pb + scan->sParam.Size.dwPixels - 1; next = -1; } else { dest = scan->UserBuf.pb; next = 1; } izoom = usb_GetScaler( scan ); ddax = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; src++ ) { ddax -= _SCALER; while((ddax < 0) && (pixels > 0)) { *dest = *src; dest += next; ddax += izoom; pixels--; } } } /** */ static void usb_GrayScale16( Plustek_Device *dev ) { u_char ls; int izoom, ddax, next; u_short *dest; u_long pixels; HiLoDef *pwm; ScanDef *scan = &dev->scanning; SANE_Bool swap = usb_HostSwap(); usb_AverageGrayWord( dev); pwm = scan->Green.philo; wSum = scan->sParam.PhyDpi.x; if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; dest = scan->UserBuf.pw + scan->sParam.Size.dwPixels - 1; } else { next = 1; dest = scan->UserBuf.pw; } izoom = usb_GetScaler( scan ); ddax = 0; if( scan->dwFlag & SCANFLAG_RightAlign ) ls = Shift; else ls = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; pwm++ ) { ddax -= _SCALER; while((ddax < 0) && (pixels > 0)) { if( swap ) *dest = _PHILO2WORD(pwm) >> ls; else *dest = _PLOHI2WORD(pwm) >> ls; dest += next; ddax += izoom; pixels--; } } } /** */ static void usb_GrayScalePseudo16( Plustek_Device *dev ) { u_char *src; int izoom, ddax, next; u_short *dest, g; u_long pixels; ScanDef *scan = &dev->scanning; usb_AverageGrayByte( dev ); if( scan->sParam.bSource == SOURCE_ADF ) { next = -1; dest = scan->UserBuf.pw + scan->sParam.Size.dwPixels - 1; } else { next = 1; dest = scan->UserBuf.pw; } src = scan->Green.pb; g = (u_short)*src; izoom = usb_GetScaler( scan ); ddax = 0; for( pixels = scan->sParam.Size.dwPixels; pixels; src++ ) { ddax -= _SCALER; while((ddax < 0) && (pixels > 0)) { *dest = (g + *src) << bShift; dest += next; ddax += izoom; pixels--; } g = (u_short)*src; } } /** function to select the apropriate pixel copy function */ static void usb_GetImageProc( Plustek_Device *dev ) { ScanDef *scan = &dev->scanning; DCapsDef *sc = &dev->usbDev.Caps; HWDef *hw = &dev->usbDev.HwSetting; bShift = 0; if( scan->sParam.UserDpi.x != scan->sParam.PhyDpi.x ) { /* Pixel scaling... */ switch( scan->sParam.bDataType ) { case SCANDATATYPE_Color: if (scan->sParam.bBitDepth > 8) { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorScale16_2; DBG( _DBG_INFO, "ImageProc is: ColorScale16_2\n" ); } else { scan->pfnProcess = usb_ColorScale16; DBG( _DBG_INFO, "ImageProc is: ColorScale16\n" ); } if (scan->fGrayFromColor) { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorScaleGray16_2; DBG( _DBG_INFO, "ImageProc is: ColorScaleGray16_2\n" ); } else { scan->pfnProcess = usb_ColorScaleGray16; DBG( _DBG_INFO, "ImageProc is: ColorScaleGray16\n" ); } } } else if (scan->dwFlag & SCANFLAG_Pseudo48) { scan->pfnProcess = usb_ColorScalePseudo16; DBG( _DBG_INFO, "ImageProc is: ColorScalePseudo16\n" ); } else if (scan->fGrayFromColor) { if( usb_IsCISDevice(dev)){ if (scan->fGrayFromColor > 7 ) { scan->pfnProcess = usb_BWScaleFromColor_2; DBG( _DBG_INFO, "ImageProc is: BWScaleFromColor_2\n" ); } else { scan->pfnProcess = usb_ColorScaleGray_2; DBG( _DBG_INFO, "ImageProc is: ColorScaleGray_2\n" ); } } else { if (scan->fGrayFromColor > 7 ) { scan->pfnProcess = usb_BWScaleFromColor; DBG( _DBG_INFO, "ImageProc is: BWScaleFromColor\n" ); } else { scan->pfnProcess = usb_ColorScaleGray; DBG( _DBG_INFO, "ImageProc is: ColorScaleGray\n" ); } } } else { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorScale8_2; DBG( _DBG_INFO, "ImageProc is: ColorScale8_2\n" ); } else { scan->pfnProcess = usb_ColorScale8; DBG( _DBG_INFO, "ImageProc is: ColorScale8\n" ); } } break; case SCANDATATYPE_Gray: if (scan->sParam.bBitDepth > 8) { scan->pfnProcess = usb_GrayScale16; DBG( _DBG_INFO, "ImageProc is: GrayScale16\n" ); } else { if (scan->dwFlag & SCANFLAG_Pseudo48) { scan->pfnProcess = usb_GrayScalePseudo16; DBG( _DBG_INFO, "ImageProc is: GrayScalePseudo16\n" ); } else { scan->pfnProcess = usb_GrayScale8; DBG( _DBG_INFO, "ImageProc is: GrayScale8\n" ); } } break; default: scan->pfnProcess = usb_BWScale; DBG( _DBG_INFO, "ImageProc is: BWScale\n" ); break; } } else { /* Pixel copy */ switch( scan->sParam.bDataType ) { case SCANDATATYPE_Color: if (scan->sParam.bBitDepth > 8) { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorDuplicate16_2; DBG( _DBG_INFO, "ImageProc is: ColorDuplicate16_2\n" ); } else { scan->pfnProcess = usb_ColorDuplicate16; DBG( _DBG_INFO, "ImageProc is: ColorDuplicate16\n" ); } if (scan->fGrayFromColor) { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorDuplicateGray16_2; DBG( _DBG_INFO, "ImageProc is: ColorDuplicateGray16_2\n" ); } else { scan->pfnProcess = usb_ColorDuplicateGray16; DBG( _DBG_INFO, "ImageProc is: ColorDuplicateGray16\n" ); } } } else if (scan->dwFlag & SCANFLAG_Pseudo48) { scan->pfnProcess = usb_ColorDuplicatePseudo16; DBG( _DBG_INFO, "ImageProc is: ColorDuplicatePseudo16\n" ); } else if (scan->fGrayFromColor) { if( usb_IsCISDevice(dev)){ if (scan->fGrayFromColor > 7 ) { scan->pfnProcess = usb_BWDuplicateFromColor_2; DBG( _DBG_INFO, "ImageProc is: BWDuplicateFromColor_2\n" ); } else { scan->pfnProcess = usb_ColorDuplicateGray_2; DBG( _DBG_INFO, "ImageProc is: ColorDuplicateGray_2\n" ); } } else { if (scan->fGrayFromColor > 7 ) { scan->pfnProcess = usb_BWDuplicateFromColor; DBG( _DBG_INFO, "ImageProc is: BWDuplicateFromColor\n" ); } else { scan->pfnProcess = usb_ColorDuplicateGray; DBG( _DBG_INFO, "ImageProc is: ColorDuplicateGray\n" ); } } } else { if( usb_IsCISDevice(dev)){ scan->pfnProcess = usb_ColorDuplicate8_2; DBG( _DBG_INFO, "ImageProc is: ColorDuplicate8_2\n" ); } else { scan->pfnProcess = usb_ColorDuplicate8; DBG( _DBG_INFO, "ImageProc is: ColorDuplicate8\n" ); } } break; case SCANDATATYPE_Gray: if (scan->sParam.bBitDepth > 8) { scan->pfnProcess = usb_GrayDuplicate16; DBG( _DBG_INFO, "ImageProc is: GrayDuplicate16\n" ); } else { if (scan->dwFlag & SCANFLAG_Pseudo48) { scan->pfnProcess = usb_GrayDuplicatePseudo16; DBG( _DBG_INFO, "ImageProc is: GrayDuplicatePseudo16\n" ); } else { scan->pfnProcess = usb_GrayDuplicate8; DBG( _DBG_INFO, "ImageProc is: GrayDuplicate8\n" ); } } break; default: scan->pfnProcess = usb_BWDuplicate; DBG( _DBG_INFO, "ImageProc is: BWDuplicate\n" ); break; } } if( scan->sParam.bBitDepth == 8 ) { if( scan->dwFlag & SCANFLAG_Pseudo48 ) { if( scan->dwFlag & SCANFLAG_RightAlign ) { bShift = 5; } else { /* this should fix the Bearpaw/U12 discrepancy * in general the fix is needed, but not for the U12 * why? - no idea! */ if(_WAF_BSHIFT7_BUG == (_WAF_BSHIFT7_BUG & sc->workaroundFlag)) bShift = 0; /* Holger Bischof 16.12.2001 */ else bShift = 7; } DBG( _DBG_INFO, "bShift adjusted: %u\n", bShift ); } } if( _LM9833 == hw->chip ) { Shift = 0; Mask = 0xFFFF; } else { Shift = 2; Mask = 0xFFFC; } } /** * here we read the image data into our intermediate buffer (in the NT version * the function was implemented as thread) */ static SANE_Int usb_ReadData( Plustek_Device *dev ) { u_long dw, dwRet, dwBytes, pl; ScanDef *scan = &dev->scanning; HWDef *hw = &dev->usbDev.HwSetting; DBG( _DBG_READ, "usb_ReadData()\n" ); pl = dev->usbDev.a_bRegs[0x4e] * hw->wDRAMSize/128; while( scan->sParam.Size.dwTotalBytes ) { if( usb_IsEscPressed()) { DBG( _DBG_INFO, "usb_ReadData() - Cancel detected...\n" ); return 0; } if( scan->sParam.Size.dwTotalBytes > scan->dwBytesScanBuf ) dw = scan->dwBytesScanBuf; else dw = scan->sParam.Size.dwTotalBytes; scan->sParam.Size.dwTotalBytes -= dw; if(!scan->sParam.Size.dwTotalBytes && dw < (pl * 1024)) { if(!(dev->usbDev.a_bRegs[0x4e] = (u_char)ceil((double)dw / (4.0 * hw->wDRAMSize)))) { dev->usbDev.a_bRegs[0x4e] = 1; } dev->usbDev.a_bRegs[0x4f] = 0; sanei_lm983x_write( dev->fd, 0x4e, &dev->usbDev.a_bRegs[0x4e], 2, SANE_TRUE ); } while( scan->bLinesToSkip ) { DBG( _DBG_READ, "Skipping %u lines\n", scan->bLinesToSkip ); dwBytes = scan->bLinesToSkip * scan->sParam.Size.dwPhyBytes; if (dwBytes > scan->dwBytesScanBuf) { dwBytes = scan->dwBytesScanBuf; scan->bLinesToSkip -= scan->dwLinesScanBuf; } else { scan->bLinesToSkip = 0; } if( !usb_ScanReadImage( dev, scan->pbGetDataBuf, dwBytes )) return 0; } if( usb_ScanReadImage( dev, scan->pbGetDataBuf, dw )) { dumpPic("plustek-pic.raw", scan->pbGetDataBuf, dw, 0); if( scan->dwLinesDiscard ) { DBG(_DBG_READ, "Discarding %lu lines\n", scan->dwLinesDiscard); dwRet = dw / scan->sParam.Size.dwPhyBytes; if (scan->dwLinesDiscard > dwRet) { scan->dwLinesDiscard -= dwRet; dwRet = 0; } else { dwRet -= scan->dwLinesDiscard; scan->dwLinesDiscard = 0; } } else { dwRet = dw / scan->sParam.Size.dwPhyBytes; } scan->pbGetDataBuf += scan->dwBytesScanBuf; if( scan->pbGetDataBuf >= scan->pbScanBufEnd ) { scan->pbGetDataBuf = scan->pbScanBufBegin; } if( dwRet ) return dwRet; } } return 0; } /* END PLUSTEK-USBIMG.C .....................................................*/