diff options
Diffstat (limited to 'backend/plustek-usbcal.c')
-rw-r--r-- | backend/plustek-usbcal.c | 1388 |
1 files changed, 1388 insertions, 0 deletions
diff --git a/backend/plustek-usbcal.c b/backend/plustek-usbcal.c new file mode 100644 index 0000000..f2e7608 --- /dev/null +++ b/backend/plustek-usbcal.c @@ -0,0 +1,1388 @@ +/*............................................................................. + * Project : SANE library for Plustek flatbed scanners; canoscan calibration + *............................................................................. + */ + +/** @file plustek-usbcal.c + * @brief Calibration routines for CanoScan CIS devices. + * + * Based on sources acquired from Plustek Inc.<br> + * Copyright (C) 2001-2007 Gerhard Jaeger <gerhard@gjaeger.de><br> + * Large parts Copyright (C) 2003 Christopher Montgomery <monty@xiph.org> + * + * Montys' comment: + * The basic premise: The stock Plustek-usbshading.c in the plustek + * driver is effectively nonfunctional for Canon CanoScan scanners. + * These scanners rely heavily on all calibration steps, especially + * fine white, to produce acceptible scan results. However, to make + * autocalibration work and make it work well involves some + * substantial mucking aobut in code that supports thirty other + * scanners with widely varying characteristics... none of which I own + * or can test. + * + * Therefore, I'm splitting out a few calibration functions I need + * to modify for the CanoScan which allows me to simplify things + * greatly for the CanoScan without worrying about breaking other + * scanners, as well as reuse the vast majority of the Plustek + * driver infrastructure without forking. + * + * History: + * - 0.45m - birth of the file; tested extensively with the LiDE 20 + * - 0.46 - renamed to plustek-usbcal.c + * - fixed problems with LiDE30, works now with 650, 1220, 670, 1240 + * - cleanup + * - added CCD calibration capability + * - added the usage of the swGain and swOffset values, to allow + * tweaking the calibration results on a sensor base + * - 0.47 - moved usb_HostSwap() to plustek_usbhw.c + * - fixed problem in cano_AdjustLightsource(), so that it won't + * stop too early. + * - 0.48 - cleanup + * - 0.49 - a_bRegs is now part of the device structure + * - fixed lampsetting in cano_AdjustLightsource() + * - 0.50 - tried to use the settings from SANE-1.0.13 + * - added _TWEAK_GAIN to allow increasing GAIN during + * lamp coarse calibration + * - added also speedtest + * - fixed segfault in fine calibration + * - 0.51 - added fine calibration cache + * - usb_SwitchLamp() now really switches off the sensor + * - 0.52 - fixed setting for frontend values (gain/offset) + * - added 0 pixel detection for offset calculation + * + * 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> + */ + +/* un-/comment the following to en-/disable lamp coarse calibration to tweak + * the initial AFE gain settings + */ +#define _TWEAK_GAIN 1 + +/* set the threshold for 0 pixels (in percent if pixels per line) */ +#define _DARK_TGT_THRESH 1 + +/** 0 for not ready, 1 pos white lamp on, 2 lamp off */ +static int strip_state = 0; + +/** depending on the strip state, the sensor is moved to the shading position + * and the lamp ist switched on + */ +static int +cano_PrepareToReadWhiteCal( Plustek_Device *dev, SANE_Bool mv2shading_pos ) +{ + SANE_Bool goto_shading_pos = SANE_TRUE; + HWDef *hw = &dev->usbDev.HwSetting; + + switch (strip_state) { + case 0: + if( !usb_IsSheetFedDevice(dev)) { + if(!usb_ModuleToHome( dev, SANE_TRUE )) { + DBG( _DBG_ERROR, "cano_PrepareToReadWhiteCal() failed\n" ); + return _E_LAMP_NOT_IN_POS; + } + } else { + goto_shading_pos = mv2shading_pos; + } + + if( goto_shading_pos ) { + if( !usb_ModuleMove(dev, MOVE_Forward, + (u_long)dev->usbDev.pSource->ShadingOriginY)) { + DBG( _DBG_ERROR, "cano_PrepareToReadWhiteCal() failed\n" ); + return _E_LAMP_NOT_IN_POS; + } + } + break; + case 2: + dev->usbDev.a_bRegs[0x29] = hw->bReg_0x29; + usb_switchLamp( dev, SANE_TRUE ); + if( !usbio_WriteReg( dev->fd, 0x29, dev->usbDev.a_bRegs[0x29])) { + DBG( _DBG_ERROR, "cano_PrepareToReadWhiteCal() failed\n" ); + return _E_LAMP_NOT_IN_POS; + } + break; + } + + strip_state = 1; + return 0; +} + +/** also here, depending on the strip state, the sensor will be moved to + * the shading position and the lamp will be switched off + */ +static int +cano_PrepareToReadBlackCal( Plustek_Device *dev ) +{ + if( strip_state == 0 ) + if(cano_PrepareToReadWhiteCal(dev, SANE_FALSE)) + return SANE_FALSE; + + if( strip_state != 2 ) { + /* + * if we have a dark shading strip, there's no need to switch + * the lamp off, leave in on a go to that strip + */ + if( dev->usbDev.pSource->DarkShadOrgY >= 0 ) { + + if( !usb_IsSheetFedDevice(dev)) + usb_ModuleToHome( dev, SANE_TRUE ); + usb_ModuleMove ( dev, MOVE_Forward, + (u_long)dev->usbDev.pSource->DarkShadOrgY ); + dev->usbDev.a_bRegs[0x45] &= ~0x10; + strip_state = 0; + + } else { + /* switch lamp off to read dark data... */ + dev->usbDev.a_bRegs[0x29] = 0; + usb_switchLamp( dev, SANE_FALSE ); + strip_state = 2; + } + } + return 0; +} + +/** according to the strip-state we switch the lamp on + */ +static int +cano_LampOnAfterCalibration( Plustek_Device *dev ) +{ + HWDef *hw = &dev->usbDev.HwSetting; + + switch (strip_state) { + case 2: + dev->usbDev.a_bRegs[0x29] = hw->bReg_0x29; + usb_switchLamp( dev, SANE_TRUE ); + if( !usbio_WriteReg( dev->fd, 0x29, dev->usbDev.a_bRegs[0x29])) { + DBG( _DBG_ERROR, "cano_LampOnAfterCalibration() failed\n" ); + return _E_LAMP_NOT_IN_POS; + } + strip_state = 1; + break; + } + return 0; +} + +/** function to adjust the CIS lamp-off setting for a given channel. + * @param min - pointer to the min OFF point for the CIS-channel + * @param max - pointer to the max OFF point for the CIS-channel + * @param off - pointer to the current OFF point of the CIS-channel + * @param val - current value to check + * @return returns 0 if the value is fine, 1, if we need to adjust + */ +static int +cano_adjLampSetting( u_short *min, u_short *max, u_short *off, u_short val ) +{ + u_long newoff = *off; + + /* perfect value, no need to adjust + * val [53440..61440] is perfect + */ + if((val < (IDEAL_GainNormal)) && (val > (IDEAL_GainNormal-8000))) + return 0; + + if(val >= (IDEAL_GainNormal-4000)) { + DBG(_DBG_INFO2, "* TOO BRIGHT --> reduce\n" ); + *max = newoff; + *off = ((newoff + *min)>>1); + + } else { + + u_short bisect = (newoff + *max)>>1; + u_short twice = newoff*2; + + DBG(_DBG_INFO2, "* TOO DARK --> up\n" ); + *min = newoff; + *off = twice<bisect?twice:bisect; + + /* as we have already set the maximum value, there's no need + * for this channel to recalibrate. + */ + if( *off > 0x3FFF ) { + DBG( _DBG_INFO, "* lamp off limited (0x%04x --> 0x3FFF)\n", *off); + *off = 0x3FFF; + return 10; + } + } + if((*min+1) >= *max ) + return 0; + + return 1; +} + +/** cano_AdjustLightsource + * coarse calibration step 0 + * [Monty changes]: On the CanoScan at least, the default lamp + * settings are several *hundred* percent too high and vary from + * scanner-to-scanner by 20-50%. This is only for CIS devices + * where the lamp_off parameter is adjustable; I'd make it more general, + * but I only have the CIS hardware to test. + */ +static int +cano_AdjustLightsource( Plustek_Device *dev ) +{ + char tmp[40]; + int i; + int res_r, res_g, res_b; + u_long dw, dwR, dwG, dwB, dwDiv, dwLoop1, dwLoop2; + RGBUShortDef max_rgb, min_rgb, tmp_rgb; + u_long *scanbuf = dev->scanning.pScanBuffer; + DCapsDef *scaps = &dev->usbDev.Caps; + HWDef *hw = &dev->usbDev.HwSetting; + + if( usb_IsEscPressed()) + return SANE_FALSE; + + DBG( _DBG_INFO, "cano_AdjustLightsource()\n" ); + + if( !usb_IsCISDevice(dev)) { + DBG( _DBG_INFO, "- function skipped, CCD device!\n" ); + + /* HEINER: we might have to tweak the PWM for the lamps */ + return SANE_TRUE; + } + + /* define the strip to scan for coarse calibration + * done at optical resolution. + */ + m_ScanParam.Size.dwLines = 1; + m_ScanParam.Size.dwPixels = scaps->Normal.Size.x * + scaps->OpticDpi.x / 300UL; + + m_ScanParam.Size.dwBytes = m_ScanParam.Size.dwPixels * 2; + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) + m_ScanParam.Size.dwBytes *=3; + + m_ScanParam.Origin.x = (u_short)((u_long) hw->wActivePixelsStart * + 300UL / scaps->OpticDpi.x); + m_ScanParam.bCalibration = PARAM_Gain; + + DBG( _DBG_INFO2, "* Coarse Calibration Strip:\n" ); + DBG( _DBG_INFO2, "* Lines = %lu\n", m_ScanParam.Size.dwLines ); + DBG( _DBG_INFO2, "* Pixels = %lu\n", m_ScanParam.Size.dwPixels ); + DBG( _DBG_INFO2, "* Bytes = %lu\n", m_ScanParam.Size.dwBytes ); + DBG( _DBG_INFO2, "* Origin.X = %u\n", m_ScanParam.Origin.x ); + + /* init... */ + max_rgb.Red = max_rgb.Green = max_rgb.Blue = 0x3fff; + min_rgb.Red = hw->red_lamp_on; + min_rgb.Green = hw->green_lamp_on; + min_rgb.Blue = hw->blue_lamp_on; + + if((dev->adj.rlampoff != -1) && + (dev->adj.glampoff != -1) && (dev->adj.rlampoff != -1)) { + DBG( _DBG_INFO, "- function skipped, using frontend values!\n" ); + return SANE_TRUE; + } + + /* we probably should preset gain to some reasonably good value + * i.e. 0x0a as it's done by Canon within their Windoze driver! + */ +#ifdef _TWEAK_GAIN + for( i=0x3b; i<0x3e; i++ ) + dev->usbDev.a_bRegs[i] = 0x0a; +#endif + for( i = 0; ; i++ ) { + + m_ScanParam.dMCLK = dMCLK; + if( !usb_SetScanParameters( dev, &m_ScanParam )) { + return SANE_FALSE; + } + + if( !usb_ScanBegin( dev, SANE_FALSE) || + !usb_ScanReadImage( dev, scanbuf, m_ScanParam.Size.dwPhyBytes ) || + !usb_ScanEnd( dev )) { + DBG( _DBG_ERROR, "* cano_AdjustLightsource() failed\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "* PhyBytes = %lu\n", m_ScanParam.Size.dwPhyBytes ); + DBG( _DBG_INFO2, "* PhyPixels = %lu\n", m_ScanParam.Size.dwPhyPixels); + + sprintf( tmp, "coarse-lamp-%u.raw", i ); + + dumpPicInit(&m_ScanParam, tmp); + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwPhyBytes, 0); + + if(usb_HostSwap()) + usb_Swap((u_short *)scanbuf, m_ScanParam.Size.dwPhyBytes ); + + sprintf( tmp, "coarse-lamp-swap%u.raw", i ); + + dumpPicInit(&m_ScanParam, tmp); + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwPhyBytes, 0); + + dwDiv = 10; + dwLoop1 = m_ScanParam.Size.dwPhyPixels/dwDiv; + + tmp_rgb.Red = tmp_rgb.Green = tmp_rgb.Blue = 0; + + /* find out the max pixel value for R, G, B */ + for( dw = 0; dwLoop1; dwLoop1-- ) { + + /* do some averaging... */ + for (dwLoop2 = dwDiv, dwR = dwG = dwB = 0; dwLoop2; dwLoop2--, dw++) { + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + + if( usb_IsCISDevice(dev)) { + dwR += ((u_short*)scanbuf)[dw]; + dwG += ((u_short*)scanbuf) + [dw+m_ScanParam.Size.dwPhyPixels+1]; + dwB += ((u_short*)scanbuf) + [dw+(m_ScanParam.Size.dwPhyPixels+1)*2]; + } else { + dwR += ((RGBUShortDef*)scanbuf)[dw].Red; + dwG += ((RGBUShortDef*)scanbuf)[dw].Green; + dwB += ((RGBUShortDef*)scanbuf)[dw].Blue; + } + } else { + dwG += ((u_short*)scanbuf)[dw]; + } + } + + dwR = dwR / dwDiv; + dwG = dwG / dwDiv; + dwB = dwB / dwDiv; + + if( tmp_rgb.Red < dwR ) + tmp_rgb.Red = dwR; + if( tmp_rgb.Green < dwG ) + tmp_rgb.Green = dwG; + if( tmp_rgb.Blue < dwB ) + tmp_rgb.Blue = dwB; + } + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + DBG( _DBG_INFO2, "red_lamp_off = %u/%u/%u\n", + min_rgb.Red ,hw->red_lamp_off, max_rgb.Red ); + } + + DBG( _DBG_INFO2, "green_lamp_off = %u/%u/%u\n", + min_rgb.Green, hw->green_lamp_off, max_rgb.Green ); + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + DBG( _DBG_INFO2, "blue_lamp_off = %u/%u/%u\n", + min_rgb.Blue, hw->blue_lamp_off, max_rgb.Blue ); + } + + DBG(_DBG_INFO2, "CUR(R,G,B)= 0x%04x(%u), 0x%04x(%u), 0x%04x(%u)\n", + tmp_rgb.Red, tmp_rgb.Red, tmp_rgb.Green, + tmp_rgb.Green, tmp_rgb.Blue, tmp_rgb.Blue ); + res_r = 0; + res_g = 0; + res_b = 0; + + /* bisect */ + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + res_r = cano_adjLampSetting( &min_rgb.Red, &max_rgb.Red, + &hw->red_lamp_off, tmp_rgb.Red ); + res_b = cano_adjLampSetting( &min_rgb.Blue, &max_rgb.Blue, + &hw->blue_lamp_off,tmp_rgb.Blue ); + } + + res_g = cano_adjLampSetting( &min_rgb.Green, &max_rgb.Green, + &hw->green_lamp_off, tmp_rgb.Green ); + + /* nothing adjusted, so stop here */ + if((res_r == 0) && (res_g == 0) && (res_b == 0)) + break; + + /* no need to adjust more, we have already reached the limit + * without tweaking the gain. + */ + if((res_r == 10) && (res_g == 10) && (res_b == 10)) + break; + + /* we raise the gain for channels, that have been limited */ +#ifdef _TWEAK_GAIN + if( res_r == 10 ) { + if( dev->usbDev.a_bRegs[0x3b] < 0xf) + dev->usbDev.a_bRegs[0x3b]++; + } + if( res_g == 10 ) { + if( dev->usbDev.a_bRegs[0x3c] < 0x0f) + dev->usbDev.a_bRegs[0x3c]++; + } + if( res_b == 10 ) { + if( dev->usbDev.a_bRegs[0x3d] < 0x0f) + dev->usbDev.a_bRegs[0x3d]++; + } +#endif + + /* now decide what to do: + * if we were too bright, we have to rerun the loop in any + * case + * if we're too dark, we should rerun it too, but we can + * compensate that using higher gain values later + */ + if( i >= 10 ) { + DBG(_DBG_INFO, "* 10 times limit reached, still too dark!!!\n"); + break; + } + usb_AdjustLamps(dev, SANE_TRUE); + } + + DBG( _DBG_INFO, "* red_lamp_on = %u\n", hw->red_lamp_on ); + DBG( _DBG_INFO, "* red_lamp_off = %u\n", hw->red_lamp_off ); + DBG( _DBG_INFO, "* green_lamp_on = %u\n", hw->green_lamp_on ); + DBG( _DBG_INFO, "* green_lamp_off = %u\n", hw->green_lamp_off ); + DBG( _DBG_INFO, "* blue_lamp_on = %u\n", hw->blue_lamp_on ); + DBG( _DBG_INFO, "* blue_lamp_off = %u\n", hw->blue_lamp_off ); + + DBG( _DBG_INFO, "cano_AdjustLightsource() done.\n" ); + return SANE_TRUE; +} + +/** + */ +static int +cano_adjGainSetting( u_char *min, u_char *max, u_char *gain,u_long val ) +{ + u_long newgain = *gain; + + if((val < IDEAL_GainNormal) && (val > (IDEAL_GainNormal-8000))) + return 0; + + if(val > (IDEAL_GainNormal-4000)) { + *max = newgain; + *gain = (newgain + *min)>>1; + } else { + *min = newgain; + *gain = (newgain + *max)>>1; + } + + if((*min+1) >= *max) + return 0; + + return 1; +} + +/** cano_AdjustGain + * function to perform the "coarse calibration step" part 1. + * We scan reference image pixels to determine the optimum coarse gain settings + * for R, G, B. (Analog gain and offset prior to ADC). These coefficients are + * applied at the line rate during normal scanning. + * The scanned line should contain a white strip with some black at the + * beginning. The function searches for the maximum value which corresponds to + * the maximum white value. + * Affects register 0x3b, 0x3c and 0x3d + * + * adjLightsource, above, steals most of this function's thunder. + */ +static SANE_Bool +cano_AdjustGain( Plustek_Device *dev ) +{ + char tmp[40]; + int i = 0, adj = 1; + u_long dw; + u_long *scanbuf = dev->scanning.pScanBuffer; + DCapsDef *scaps = &dev->usbDev.Caps; + HWDef *hw = &dev->usbDev.HwSetting; + + unsigned char max[3], min[3]; + + if( usb_IsEscPressed()) + return SANE_FALSE; + + bMaxITA = 0xff; + + max[0] = max[1] = max[2] = 0x3f; + min[0] = min[1] = min[2] = 1; + + DBG( _DBG_INFO, "cano_AdjustGain()\n" ); + if( !usb_InCalibrationMode(dev)) { + if((dev->adj.rgain != -1) && + (dev->adj.ggain != -1) && (dev->adj.bgain != -1)) { + setAdjGain( dev->adj.rgain, &dev->usbDev.a_bRegs[0x3b] ); + setAdjGain( dev->adj.ggain, &dev->usbDev.a_bRegs[0x3c] ); + setAdjGain( dev->adj.bgain, &dev->usbDev.a_bRegs[0x3d] ); + DBG( _DBG_INFO, "- function skipped, using frontend values!\n" ); + return SANE_TRUE; + } + } + + /* define the strip to scan for coarse calibration + * done at 300dpi + */ + m_ScanParam.Size.dwLines = 1; /* for gain */ + m_ScanParam.Size.dwPixels = scaps->Normal.Size.x * + scaps->OpticDpi.x / 300UL; + + m_ScanParam.Size.dwBytes = m_ScanParam.Size.dwPixels * 2; + + if( usb_IsCISDevice(dev) && m_ScanParam.bDataType == SCANDATATYPE_Color) + m_ScanParam.Size.dwBytes *=3; + + m_ScanParam.Origin.x = (u_short)((u_long) hw->wActivePixelsStart * + 300UL / scaps->OpticDpi.x); + m_ScanParam.bCalibration = PARAM_Gain; + + DBG( _DBG_INFO2, "Coarse Calibration Strip:\n" ); + DBG( _DBG_INFO2, "Lines = %lu\n", m_ScanParam.Size.dwLines ); + DBG( _DBG_INFO2, "Pixels = %lu\n", m_ScanParam.Size.dwPixels ); + DBG( _DBG_INFO2, "Bytes = %lu\n", m_ScanParam.Size.dwBytes ); + DBG( _DBG_INFO2, "Origin.X = %u\n", m_ScanParam.Origin.x ); + + while( adj ) { + + m_ScanParam.dMCLK = dMCLK; + + if( !usb_SetScanParameters( dev, &m_ScanParam )) + return SANE_FALSE; + + if( !usb_ScanBegin( dev, SANE_FALSE) || + !usb_ScanReadImage(dev,scanbuf,m_ScanParam.Size.dwPhyBytes) || + !usb_ScanEnd( dev )) { + DBG( _DBG_ERROR, "cano_AdjustGain() failed\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "PhyBytes = %lu\n", m_ScanParam.Size.dwPhyBytes ); + DBG( _DBG_INFO2, "PhyPixels = %lu\n", m_ScanParam.Size.dwPhyPixels ); + + sprintf( tmp, "coarse-gain-%u.raw", i++ ); + + dumpPicInit(&m_ScanParam, tmp); + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwPhyBytes, 0); + + if(usb_HostSwap()) + usb_Swap((u_short *)scanbuf, m_ScanParam.Size.dwPhyBytes ); + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + + RGBUShortDef max_rgb; + u_long dwR, dwG, dwB; + u_long dwDiv = 10; + u_long dwLoop1 = m_ScanParam.Size.dwPhyPixels/dwDiv, dwLoop2; + + max_rgb.Red = max_rgb.Green = max_rgb.Blue = 0; + + /* find out the max pixel value for R, G, B */ + for( dw = 0; dwLoop1; dwLoop1-- ) { + + /* do some averaging... */ + for (dwLoop2 = dwDiv, dwR=dwG=dwB=0; dwLoop2; dwLoop2--, dw++) { + + if( usb_IsCISDevice(dev)) { + dwR += ((u_short*)scanbuf)[dw]; + dwG += ((u_short*)scanbuf) + [dw+m_ScanParam.Size.dwPhyPixels+1]; + dwB += ((u_short*)scanbuf) + [dw+(m_ScanParam.Size.dwPhyPixels+1)*2]; + } else { + dwR += ((RGBUShortDef*)scanbuf)[dw].Red; + dwG += ((RGBUShortDef*)scanbuf)[dw].Green; + dwB += ((RGBUShortDef*)scanbuf)[dw].Blue; + } + } + dwR = dwR / dwDiv; + dwG = dwG / dwDiv; + dwB = dwB / dwDiv; + + if(max_rgb.Red < dwR) + max_rgb.Red = dwR; + if(max_rgb.Green < dwG) + max_rgb.Green = dwG; + if(max_rgb.Blue < dwB) + max_rgb.Blue = dwB; + } + + DBG(_DBG_INFO2, "MAX(R,G,B)= 0x%04x(%u), 0x%04x(%u), 0x%04x(%u)\n", + max_rgb.Red, max_rgb.Red, max_rgb.Green, + max_rgb.Green, max_rgb.Blue, max_rgb.Blue ); + + adj = cano_adjGainSetting(min , max ,dev->usbDev.a_bRegs+0x3b,max_rgb.Red ); + adj += cano_adjGainSetting(min+1, max+1,dev->usbDev.a_bRegs+0x3c,max_rgb.Green); + adj += cano_adjGainSetting(min+2, max+2,dev->usbDev.a_bRegs+0x3d,max_rgb.Blue ); + + } else { + + u_short w_max = 0; + + for( dw = 0; dw < m_ScanParam.Size.dwPhyPixels; dw++ ) { + if( w_max < ((u_short*)scanbuf)[dw]) + w_max = ((u_short*)scanbuf)[dw]; + } + + adj = cano_adjGainSetting(min,max,dev->usbDev.a_bRegs+0x3c,w_max); + dev->usbDev.a_bRegs[0x3b] = (dev->usbDev.a_bRegs[0x3d] = dev->usbDev.a_bRegs[0x3c]); + + DBG(_DBG_INFO2, "MAX(G)= 0x%04x(%u)\n", w_max, w_max ); + + } + DBG( _DBG_INFO2, "REG[0x3b] = %u\n", dev->usbDev.a_bRegs[0x3b] ); + DBG( _DBG_INFO2, "REG[0x3c] = %u\n", dev->usbDev.a_bRegs[0x3c] ); + DBG( _DBG_INFO2, "REG[0x3d] = %u\n", dev->usbDev.a_bRegs[0x3d] ); + } + DBG( _DBG_INFO, "cano_AdjustGain() done.\n" ); + return SANE_TRUE; +} + +static int tweak_offset[3]; + +/** + */ +static int +cano_GetNewOffset(Plustek_Device *dev, u_long *val, int channel, signed char *low, + signed char *now, signed char *high, u_long *zc) +{ + DCapsDef *scaps = &dev->usbDev.Caps; + + if (tweak_offset[channel]) { + + /* if we're too black, we're likely off the low end */ + if( val[channel] <= 16 ) { + low[channel] = now[channel]; + now[channel] = (now[channel]+high[channel])/2; + + dev->usbDev.a_bRegs[0x38+channel]= (now[channel]&0x3f); + + if( low[channel]+1 >= high[channel] ) + return 0; + return 1; + + } else if ( val[channel]>=2048 ) { + high[channel]=now[channel]; + now[channel]=(now[channel]+low[channel])/2; + + dev->usbDev.a_bRegs[0x38+channel]= (now[channel]&0x3f); + + if(low[channel]+1>=high[channel]) + return 0; + return 1; + } + } + + if (!(scaps->workaroundFlag & _WAF_INC_DARKTGT)) { + DBG( _DBG_INFO, "0 Pixel adjustment not active!\n"); + return 0; + } + + /* reaching this point, our black level should be okay, but + * we also should check the percentage of 0 level pixels. + * It turned out, that when having a lot of 0 level pixels, + * the calibration will be bad and the resulting scans show up + * stripes... + */ + if (zc[channel] > _DARK_TGT_THRESH) { + DBG( _DBG_INFO2, "More than %u%% 0 pixels detected, raise offset!\n", + _DARK_TGT_THRESH); + high[channel]=now[channel]; + now[channel]=(now[channel]+low[channel])/2; + + /* no more value checks, the goal to set the black level < 2048 + * will cause stripes... + */ + tweak_offset[channel] = 0; + + dev->usbDev.a_bRegs[0x38+channel]= (now[channel]&0x3f); + + if( low[channel]+1 >= high[channel] ) + return 0; + return 1; + + } +#if 0 + else if ( val[channel]>=4096 ) { + low[channel] = now[channel]; + now[channel] = (now[channel]+high[channel])/2; + + dev->usbDev.a_bRegs[0x38+channel]= (now[channel]&0x3f); + + if( low[channel]+1 >= high[channel] ) + return 0; + return 1; + } +#endif + return 0; +} + +/** cano_AdjustOffset + * function to perform the "coarse calibration step" part 2. + * We scan reference image pixels to determine the optimum coarse offset settings + * for R, G, B. (Analog gain and offset prior to ADC). These coefficients are + * applied at the line rate during normal scanning. + * On CIS based devices, we switch the light off, on CCD devices, we use the optical + * black pixels. + * Affects register 0x38, 0x39 and 0x3a + */ + +/* Move this to a bisection-based algo and correct some fenceposts; + Plustek's example code disagrees with NatSemi's docs; going by the + docs works better, I will assume the docs are correct. --Monty */ + +static int +cano_AdjustOffset( Plustek_Device *dev ) +{ + char tmp[40]; + int i, adj; + u_short r, g, b; + u_long dw, dwPixels; + u_long dwSum[3], zCount[3]; + + signed char low[3] = {-32,-32,-32 }; + signed char now[3] = { 0, 0, 0 }; + signed char high[3] = { 31, 31, 31 }; + + u_long *scanbuf = dev->scanning.pScanBuffer; + HWDef *hw = &dev->usbDev.HwSetting; + DCapsDef *scaps = &dev->usbDev.Caps; + + if( usb_IsEscPressed()) + return SANE_FALSE; + + DBG( _DBG_INFO, "cano_AdjustOffset()\n" ); + if( !usb_InCalibrationMode(dev)) { + if((dev->adj.rofs != -1) && + (dev->adj.gofs != -1) && (dev->adj.bofs != -1)) { + dev->usbDev.a_bRegs[0x38] = (dev->adj.rofs & 0x3f); + dev->usbDev.a_bRegs[0x39] = (dev->adj.gofs & 0x3f); + dev->usbDev.a_bRegs[0x3a] = (dev->adj.bofs & 0x3f); + DBG( _DBG_INFO, "- function skipped, using frontend values!\n" ); + return SANE_TRUE; + } + } + + m_ScanParam.Size.dwLines = 1; + m_ScanParam.Size.dwPixels = scaps->Normal.Size.x*scaps->OpticDpi.x/300UL; + + if( usb_IsCISDevice(dev)) + dwPixels = m_ScanParam.Size.dwPixels; + else + dwPixels = (u_long)(hw->bOpticBlackEnd - hw->bOpticBlackStart); + + m_ScanParam.Size.dwBytes = m_ScanParam.Size.dwPixels * 2; + + if( usb_IsCISDevice(dev) && m_ScanParam.bDataType == SCANDATATYPE_Color) + m_ScanParam.Size.dwBytes *= 3; + + m_ScanParam.Origin.x = (u_short)((u_long)hw->bOpticBlackStart * 300UL / + dev->usbDev.Caps.OpticDpi.x); + m_ScanParam.bCalibration = PARAM_Offset; + m_ScanParam.dMCLK = dMCLK; + + if( !usb_SetScanParameters( dev, &m_ScanParam )) { + DBG( _DBG_ERROR, "cano_AdjustOffset() failed\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "S.dwPixels = %lu\n", m_ScanParam.Size.dwPixels ); + DBG( _DBG_INFO2, "dwPixels = %lu\n", dwPixels ); + DBG( _DBG_INFO2, "dwPhyBytes = %lu\n", m_ScanParam.Size.dwPhyBytes ); + DBG( _DBG_INFO2, "dwPhyPixels = %lu\n", m_ScanParam.Size.dwPhyPixels ); + + tweak_offset[0] = + tweak_offset[1] = + tweak_offset[2] = 1; + + for( i = 0, adj = 1; adj != 0; i++ ) { + + if((!usb_ScanBegin(dev, SANE_FALSE)) || + (!usb_ScanReadImage(dev,scanbuf,m_ScanParam.Size.dwPhyBytes)) || + !usb_ScanEnd( dev )) { + DBG( _DBG_ERROR, "cano_AdjustOffset() failed\n" ); + return SANE_FALSE; + } + + sprintf( tmp, "coarse-off-%u.raw", i ); + + dumpPicInit(&m_ScanParam, tmp); + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwPhyBytes, 0); + + if(usb_HostSwap()) + usb_Swap((u_short *)scanbuf, m_ScanParam.Size.dwPhyBytes ); + + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + + dwSum[0] = dwSum[1] = dwSum[2] = 0; + zCount[0] = zCount[1] = zCount[2] = 0; + + for (dw = 0; dw < dwPixels; dw++) { + + if( usb_IsCISDevice(dev)) { + + r = ((u_short*)scanbuf)[dw]; + g = ((u_short*)scanbuf)[dw+m_ScanParam.Size.dwPhyPixels+1]; + b = ((u_short*)scanbuf)[dw+(m_ScanParam.Size.dwPhyPixels+1)*2]; + + } else { + r = ((RGBUShortDef*)scanbuf)[dw].Red; + g = ((RGBUShortDef*)scanbuf)[dw].Green; + b = ((RGBUShortDef*)scanbuf)[dw].Blue; + } + + dwSum[0] += r; + dwSum[1] += g; + dwSum[2] += b; + + if (r==0) zCount[0]++; + if (g==0) zCount[1]++; + if (b==0) zCount[2]++; + } + + DBG( _DBG_INFO2, "RedSum = %lu, ave = %lu, ZC=%lu, %lu%%\n", + dwSum[0], dwSum[0]/dwPixels, + zCount[0], (zCount[0]*100)/dwPixels); + DBG( _DBG_INFO2, "GreenSum = %lu, ave = %lu, ZC=%lu, %lu%%\n", + dwSum[1], dwSum[1]/dwPixels, + zCount[1], (zCount[1]*100)/dwPixels); + DBG( _DBG_INFO2, "BlueSum = %lu, ave = %lu, ZC=%lu, %lu%%\n", + dwSum[2], dwSum[2]/dwPixels, + zCount[2], (zCount[2]*100)/dwPixels); + + /* do averaging for each channel */ + dwSum[0] /= dwPixels; + dwSum[1] /= dwPixels; + dwSum[2] /= dwPixels; + + zCount[0] = (zCount[0] * 100)/ dwPixels; + zCount[1] = (zCount[1] * 100)/ dwPixels; + zCount[2] = (zCount[2] * 100)/ dwPixels; + + adj = cano_GetNewOffset(dev, dwSum, 0, low, now, high, zCount); + adj |= cano_GetNewOffset(dev, dwSum, 1, low, now, high, zCount); + adj |= cano_GetNewOffset(dev, dwSum, 2, low, now, high, zCount); + + DBG( _DBG_INFO2, "RedOff = %d/%d/%d\n", + (int)low[0],(int)now[0],(int)high[0]); + DBG( _DBG_INFO2, "GreenOff = %d/%d/%d\n", + (int)low[1],(int)now[1],(int)high[1]); + DBG( _DBG_INFO2, "BlueOff = %d/%d/%d\n", + (int)low[2],(int)now[2],(int)high[2]); + + } else { + dwSum[0] = 0; + zCount[0] = 0; + + for( dw = 0; dw < dwPixels; dw++ ) { + dwSum[0] += ((u_short*)scanbuf)[dw]; + + if (((u_short*)scanbuf)[dw] == 0) + zCount[0]++; + } + + DBG( _DBG_INFO2, "Sum=%lu, ave=%lu, ZC=%lu, %lu%%\n", + dwSum[0],dwSum[0]/dwPixels, + zCount[0], (zCount[0]*100)/dwPixels); + + dwSum[0] /= dwPixels; + zCount[0] = (zCount[0] * 100)/ dwPixels; + + adj = cano_GetNewOffset(dev, dwSum, 0, low, now, high, zCount); + + dev->usbDev.a_bRegs[0x3a] = + dev->usbDev.a_bRegs[0x39] = dev->usbDev.a_bRegs[0x38]; + + DBG( _DBG_INFO2, "GrayOff = %d/%d/%d\n", + (int)low[0],(int)now[0],(int)high[0]); + } + + DBG( _DBG_INFO2, "REG[0x38] = %u\n", dev->usbDev.a_bRegs[0x38] ); + DBG( _DBG_INFO2, "REG[0x39] = %u\n", dev->usbDev.a_bRegs[0x39] ); + DBG( _DBG_INFO2, "REG[0x3a] = %u\n", dev->usbDev.a_bRegs[0x3a] ); + + _UIO(sanei_lm983x_write(dev->fd, 0x38, &dev->usbDev.a_bRegs[0x38], 3, SANE_TRUE)); + } + + /* is that really needed?! */ + if( m_ScanParam.bDataType == SCANDATATYPE_Color ) { + dev->usbDev.a_bRegs[0x38] = now[0] & 0x3f; + dev->usbDev.a_bRegs[0x39] = now[1] & 0x3f; + dev->usbDev.a_bRegs[0x3a] = now[2] & 0x3f; + } else { + dev->usbDev.a_bRegs[0x38] = + dev->usbDev.a_bRegs[0x39] = + dev->usbDev.a_bRegs[0x3a] = now[0] & 0x3f; + } + + DBG( _DBG_INFO, "cano_AdjustOffset() done.\n" ); + return SANE_TRUE; +} + +/** usb_AdjustDarkShading + * fine calibration part 1 + */ +static SANE_Bool +cano_AdjustDarkShading( Plustek_Device *dev, u_short cal_dpi ) +{ + char tmp[40]; + ScanParam *param = &dev->scanning.sParam; + ScanDef *scan = &dev->scanning; + u_long *scanbuf = scan->pScanBuffer; + u_short *bufp; + unsigned int i, j; + int step, stepW, val; + u_long red, green, blue, gray; + + DBG( _DBG_INFO, "cano_AdjustDarkShading()\n" ); + if( usb_IsEscPressed()) + return SANE_FALSE; + + usb_PrepareFineCal( dev, &m_ScanParam, cal_dpi ); + m_ScanParam.bCalibration = PARAM_DarkShading; + + sprintf( tmp, "fine-dark.raw" ); + dumpPicInit(&m_ScanParam, tmp); + + usb_SetScanParameters( dev, &m_ScanParam ); + if( usb_ScanBegin( dev, SANE_FALSE ) && + usb_ScanReadImage( dev, scanbuf, m_ScanParam.Size.dwTotalBytes)) { + + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwTotalBytes, 0); + + if(usb_HostSwap()) + usb_Swap((u_short *)scanbuf, m_ScanParam.Size.dwTotalBytes); + } + if (!usb_ScanEnd( dev )){ + DBG( _DBG_ERROR, "cano_AdjustDarkShading() failed\n" ); + return SANE_FALSE; + } + + /* average the n lines, compute reg values */ + if( scan->sParam.bDataType == SCANDATATYPE_Color ) { + + stepW = m_ScanParam.Size.dwPhyPixels; + if( usb_IsCISDevice(dev)) + step = m_ScanParam.Size.dwPhyPixels + 1; + else + step = (m_ScanParam.Size.dwPhyPixels*3) + 1; + + for( i=0; i<m_ScanParam.Size.dwPhyPixels; i++ ) { + + red = 0; + green = 0; + blue = 0; + if( usb_IsCISDevice(dev)) + bufp = ((u_short *)scanbuf)+i; + else + bufp = ((u_short *)scanbuf)+(i*3); + + for( j=0; j<m_ScanParam.Size.dwPhyLines; j++ ) { + + if( usb_IsCISDevice(dev)) { + red += *bufp; bufp+=step; + green += *bufp; bufp+=step; + blue += *bufp; bufp+=step; + } else { + + red += bufp[0]; + green += bufp[1]; + blue += bufp[2]; + + bufp += step; + } + } + + val = ((int)(red/m_ScanParam.Size.dwPhyLines) + param->swOffset[0]); + if( val < 0 ) { + DBG( _DBG_INFO, "val < 0!!!!\n" ); + val = 0; + } + a_wDarkShading[i] = (u_short)val; + + val = ((int)(green/m_ScanParam.Size.dwPhyLines) + param->swOffset[1]); + if( val < 0 ) { + DBG( _DBG_INFO, "val < 0!!!!\n" ); + val = 0; + } + a_wDarkShading[i+stepW] = (u_short)val; + + val = ((int)(blue/m_ScanParam.Size.dwPhyLines) + param->swOffset[2]); + if( val < 0 ) { + DBG( _DBG_INFO, "val < 0!!!!\n" ); + val = 0; + } + a_wDarkShading[i+stepW*2] = (u_short)val; + } + + } else { + + step = m_ScanParam.Size.dwPhyPixels + 1; + for( i=0; i<m_ScanParam.Size.dwPhyPixels; i++ ) { + + gray = 0; + bufp = ((u_short *)scanbuf)+i; + + for( j=0; j < m_ScanParam.Size.dwPhyLines; j++ ) { + gray += *bufp; + bufp += step; + } + a_wDarkShading[i]= gray/j + param->swOffset[0]; + } + + memcpy( a_wDarkShading + m_ScanParam.Size.dwPhyPixels, + a_wDarkShading, m_ScanParam.Size.dwPhyPixels * 2); + memcpy( a_wDarkShading + m_ScanParam.Size.dwPhyPixels * 2, + a_wDarkShading, m_ScanParam.Size.dwPhyPixels * 2); + } + + if(usb_HostSwap()) + usb_Swap(a_wDarkShading, m_ScanParam.Size.dwPhyPixels * 2 * 3); + + usb_line_statistics( "Dark", a_wDarkShading, m_ScanParam.Size.dwPhyPixels, + scan->sParam.bDataType == SCANDATATYPE_Color?1:0); + + DBG( _DBG_INFO, "cano_AdjustDarkShading() done\n" ); + return SANE_TRUE; +} + +/** usb_AdjustWhiteShading + * fine calibration part 2 - read the white calibration area and calculate + * the gain coefficient for each pixel + */ +static SANE_Bool +cano_AdjustWhiteShading( Plustek_Device *dev, u_short cal_dpi ) +{ + char tmp[40]; + ScanParam *param = &dev->scanning.sParam; + ScanDef *scan = &dev->scanning; + u_long *scanbuf = scan->pScanBuffer; + u_short *bufp; + unsigned int i, j; + int step, stepW; + u_long red, green, blue, gray; + + DBG( _DBG_INFO, "cano_AdjustWhiteShading()\n" ); + if( usb_IsEscPressed()) + return SANE_FALSE; + + usb_PrepareFineCal( dev, &m_ScanParam, cal_dpi ); + m_ScanParam.bCalibration = PARAM_WhiteShading; + + sprintf( tmp, "fine-white.raw" ); + DBG( _DBG_INFO2, "FINE WHITE Calibration Strip: %s\n", tmp ); + DBG( _DBG_INFO2, "Lines = %lu\n", m_ScanParam.Size.dwLines ); + DBG( _DBG_INFO2, "Pixels = %lu\n", m_ScanParam.Size.dwPixels ); + DBG( _DBG_INFO2, "Bytes = %lu\n", m_ScanParam.Size.dwBytes ); + DBG( _DBG_INFO2, "Origin.X = %u\n", m_ScanParam.Origin.x ); + dumpPicInit(&m_ScanParam, tmp); + + if( usb_SetScanParameters( dev, &m_ScanParam ) && + usb_ScanBegin( dev, SANE_FALSE ) && + usb_ScanReadImage( dev, scanbuf, m_ScanParam.Size.dwTotalBytes)) { + + dumpPic(tmp, (u_char*)scanbuf, m_ScanParam.Size.dwTotalBytes, 0); + + if(usb_HostSwap()) + usb_Swap((u_short *)scanbuf, m_ScanParam.Size.dwTotalBytes); + + if (!usb_ScanEnd( dev )) { + DBG( _DBG_ERROR, "cano_AdjustWhiteShading() failed\n" ); + return SANE_FALSE; + } + } else { + DBG( _DBG_ERROR, "cano_AdjustWhiteShading() failed\n" ); + return SANE_FALSE; + } + + /* average the n lines, compute reg values */ + if( scan->sParam.bDataType == SCANDATATYPE_Color ) { + + stepW = m_ScanParam.Size.dwPhyPixels; + if( usb_IsCISDevice(dev)) + step = m_ScanParam.Size.dwPhyPixels + 1; + else + step = (m_ScanParam.Size.dwPhyPixels*3) + 1; + + for( i=0; i < m_ScanParam.Size.dwPhyPixels; i++ ) { + + red = 0; + green = 0; + blue = 0; + if( usb_IsCISDevice(dev)) + bufp = ((u_short *)scanbuf)+i; + else + bufp = ((u_short *)scanbuf)+(i*3); + + for( j=0; j<m_ScanParam.Size.dwPhyLines; j++ ) { + + if( usb_IsCISDevice(dev)) { + red += *bufp; bufp+=step; + green += *bufp; bufp+=step; + blue += *bufp; bufp+=step; + } else { + red += bufp[0]; + green += bufp[1]; + blue += bufp[2]; + bufp += step; + } + } + + /* tweaked by the settings in swGain --> 1000/swGain[r,g,b] */ + red = (65535.*1000./(double)param->swGain[0]) * 16384.*j/red; + green = (65535.*1000./(double)param->swGain[1]) * 16384.*j/green; + blue = (65535.*1000./(double)param->swGain[2]) * 16384.*j/blue; + + a_wWhiteShading[i] = (red > 65535 ? 65535:red ); + a_wWhiteShading[i+stepW] = (green > 65535 ? 65535:green); + a_wWhiteShading[i+stepW*2] = (blue > 65535 ? 65535:blue ); + } + + } else { + + step = m_ScanParam.Size.dwPhyPixels + 1; + for( i=0; i<m_ScanParam.Size.dwPhyPixels; i++ ){ + gray = 0; + bufp = ((u_short *)scanbuf)+i; + + for( j=0; j<m_ScanParam.Size.dwPhyLines; j++ ) { + gray += *bufp; + bufp += step; + } + + gray = (65535.*1000./(double)param->swGain[0]) * 16384.*j/gray; + + a_wWhiteShading[i]= (gray > 65535 ? 65535:gray); + } + + memcpy( a_wWhiteShading + m_ScanParam.Size.dwPhyPixels, + a_wWhiteShading, m_ScanParam.Size.dwPhyPixels * 2); + memcpy( a_wWhiteShading + m_ScanParam.Size.dwPhyPixels * 2, + a_wWhiteShading, m_ScanParam.Size.dwPhyPixels * 2); + } + + if(usb_HostSwap()) + usb_Swap(a_wWhiteShading, m_ScanParam.Size.dwPhyPixels * 2 * 3 ); + + usb_SaveCalSetShading( dev, &m_ScanParam ); + + usb_line_statistics( "White", a_wWhiteShading, m_ScanParam.Size.dwPhyPixels, + scan->sParam.bDataType == SCANDATATYPE_Color?1:0); + + DBG( _DBG_INFO, "cano_AdjustWhiteShading() done\n" ); + return SANE_TRUE; +} + +/** the entry function for the CIS calibration stuff. + */ +static int +cano_DoCalibration( Plustek_Device *dev ) +{ + u_short dpi, idx, idx_end; + u_long save_waf; + SANE_Bool skip_fine; + ScanDef *scan = &dev->scanning; + HWDef *hw = &dev->usbDev.HwSetting; + DCapsDef *scaps = &dev->usbDev.Caps; + + if( SANE_TRUE == scan->fCalibrated ) + return SANE_TRUE; + + DBG( _DBG_INFO, "cano_DoCalibration()\n" ); + + if( _IS_PLUSTEKMOTOR(hw->motorModel)){ + DBG( _DBG_ERROR, "altCalibration can't work with this " + "Plustek motor control setup\n" ); + return SANE_FALSE; /* can't cal this */ + } + + /* Don't allow calibration settings from the other driver to confuse our + * use of a few of its functions. + */ + save_waf = scaps->workaroundFlag; + scaps->workaroundFlag &= ~_WAF_SKIP_WHITEFINE; + scaps->workaroundFlag &= ~_WAF_SKIP_FINE; + scaps->workaroundFlag &= ~_WAF_BYPASS_CALIBRATION; + + if( !dev->adj.cacheCalData && !usb_IsSheetFedDevice(dev)) + usb_SpeedTest( dev ); + + /* here we handle that warmup stuff for CCD devices */ + if( !usb_AutoWarmup( dev )) + return SANE_FALSE; + + /* Set the shading position to undefined */ + strip_state = 0; + usb_PrepareCalibration( dev ); + + usb_SetMCLK( dev, &scan->sParam ); + + if( !scan->skipCoarseCalib ) { + + if( !usb_Wait4ScanSample( dev )) + return SANE_FALSE; + + DBG( _DBG_INFO2, "###### ADJUST LAMP (COARSE)#######\n" ); + if( cano_PrepareToReadWhiteCal(dev, SANE_TRUE)) + return SANE_FALSE; + + dev->usbDev.a_bRegs[0x45] &= ~0x10; + if( !cano_AdjustLightsource(dev)) { + DBG( _DBG_ERROR, "Coarse Calibration failed!!!\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "###### ADJUST OFFSET (COARSE) ####\n" ); + if(cano_PrepareToReadBlackCal(dev)) + return SANE_FALSE; + + if( !cano_AdjustOffset(dev)) { + DBG( _DBG_ERROR, "Coarse Calibration failed!!!\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "###### ADJUST GAIN (COARSE)#######\n" ); + if(cano_PrepareToReadWhiteCal(dev, SANE_FALSE)) + return SANE_FALSE; + + if( !cano_AdjustGain(dev)) { + DBG( _DBG_ERROR, "Coarse Calibration failed!!!\n" ); + return SANE_FALSE; + } + } else { + strip_state = 1; + DBG( _DBG_INFO2, "###### COARSE calibration skipped #######\n" ); + } + + skip_fine = SANE_FALSE; + idx_end = 2; + if( dev->adj.cacheCalData || usb_IsSheetFedDevice(dev)) { + + skip_fine = usb_FineShadingFromFile(dev); + + /* we recalibrate in any case ! */ + if( usb_InCalibrationMode(dev)) { + skip_fine = SANE_FALSE; + idx_end = DIVIDER+1; + + /* did I say any case? */ + if (scan->sParam.bBitDepth != 8) { + skip_fine = SANE_TRUE; + DBG( _DBG_INFO2, "No fine calibration for non-8bit modes!\n" ); + } + + } else if( usb_IsSheetFedDevice(dev)) { + + /* we only do the calibration upon request !*/ + if( !skip_fine ) { + DBG( _DBG_INFO2, "SHEET-FED device, skip fine calibration!\n" ); + skip_fine = SANE_TRUE; + scaps->workaroundFlag |= _WAF_BYPASS_CALIBRATION; + } + } + } + + if( !skip_fine ) { + + for( idx = 1; idx < idx_end; idx++ ) { + + dpi = 0; + if( usb_InCalibrationMode(dev)) { + dpi = usb_get_res( scaps->OpticDpi.x, idx ); + + /* we might should check against device specific limit */ + if(dpi < 50) + continue; + } + + DBG( _DBG_INFO2, "###### ADJUST DARK (FINE) ########\n" ); + if(cano_PrepareToReadBlackCal(dev)) + return SANE_FALSE; + + dev->usbDev.a_bRegs[0x45] |= 0x10; + if( !cano_AdjustDarkShading(dev, dpi)) { + DBG( _DBG_ERROR, "Fine Calibration failed!!!\n" ); + return SANE_FALSE; + } + + DBG( _DBG_INFO2, "###### ADJUST WHITE (FINE) #######\n" ); + if(cano_PrepareToReadWhiteCal(dev, SANE_FALSE)) + return SANE_FALSE; + + if( !usb_IsSheetFedDevice(dev)) { + if(!usb_ModuleToHome( dev, SANE_TRUE )) + return SANE_FALSE; + + if( !usb_ModuleMove(dev, MOVE_Forward, + (u_long)dev->usbDev.pSource->ShadingOriginY)) { + return SANE_FALSE; + } + } + if( !cano_AdjustWhiteShading(dev, dpi)) { + DBG( _DBG_ERROR, "Fine Calibration failed!!!\n" ); + return SANE_FALSE; + } + + /* force to go back */ + strip_state = 0; + } + } else { + DBG( _DBG_INFO2, "###### FINE calibration skipped #######\n" ); + + dev->usbDev.a_bRegs[0x45] |= 0x10; + strip_state = 2; + + m_ScanParam = scan->sParam; + usb_GetPhyPixels( dev, &m_ScanParam ); + + usb_line_statistics( "Dark", a_wDarkShading, m_ScanParam.Size.dwPhyPixels, + m_ScanParam.bDataType == SCANDATATYPE_Color?1:0); + usb_line_statistics( "White", a_wWhiteShading, m_ScanParam.Size.dwPhyPixels, + m_ScanParam.bDataType == SCANDATATYPE_Color?1:0); + } + + /* Lamp on if it's not */ + cano_LampOnAfterCalibration(dev); + strip_state = 0; + + /* home the sensor after calibration + */ + if( !usb_IsSheetFedDevice(dev)) + usb_ModuleToHome( dev, SANE_TRUE ); + scan->fCalibrated = SANE_TRUE; + + DBG( _DBG_INFO, "cano_DoCalibration() done\n" ); + DBG( _DBG_INFO, "-------------------------\n" ); + DBG( _DBG_INFO, "Static Gain:\n" ); + DBG( _DBG_INFO, "REG[0x3b] = %u\n", dev->usbDev.a_bRegs[0x3b] ); + DBG( _DBG_INFO, "REG[0x3c] = %u\n", dev->usbDev.a_bRegs[0x3c] ); + DBG( _DBG_INFO, "REG[0x3d] = %u\n", dev->usbDev.a_bRegs[0x3d] ); + DBG( _DBG_INFO, "Static Offset:\n" ); + DBG( _DBG_INFO, "REG[0x38] = %i\n", dev->usbDev.a_bRegs[0x38] ); + DBG( _DBG_INFO, "REG[0x39] = %i\n", dev->usbDev.a_bRegs[0x39] ); + DBG( _DBG_INFO, "REG[0x3a] = %i\n", dev->usbDev.a_bRegs[0x3a] ); + DBG( _DBG_INFO, "-------------------------\n" ); + + scaps->workaroundFlag |= save_waf; + + return SANE_TRUE; +} + +/* END PLUSTEK-USBCAL.C .....................................................*/ |