kopia lustrzana https://gitlab.com/sane-project/backends
				
				
				
			
		
			
				
	
	
		
			1292 wiersze
		
	
	
		
			39 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			1292 wiersze
		
	
	
		
			39 KiB
		
	
	
	
		
			C
		
	
	
| /*.............................................................................
 | |
|  * 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-2006 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
 | |
|  *
 | |
|  * 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
 | |
| 
 | |
| /** 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_char      *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, scanbuf, m_ScanParam.Size.dwPhyBytes );
 | |
| 
 | |
| 		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, scanbuf, m_ScanParam.Size.dwPhyBytes );
 | |
| 
 | |
| 		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_char   *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((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, scanbuf, m_ScanParam.Size.dwPhyBytes );
 | |
| 
 | |
| 		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
 | |
| cano_GetNewOffset( Plustek_Device *dev, u_long *val, int channel,
 | |
|                    signed char *low, signed char *now, signed char *high )
 | |
| {
 | |
| 	/* 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;
 | |
| 	}
 | |
| 	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];
 | |
| 
 | |
| 	signed char low[3]  = {-32,-32,-32 };
 | |
| 	signed char now[3]  = {  0,  0,  0 };
 | |
| 	signed char high[3] = { 31, 31, 31 };
 | |
| 
 | |
| 	u_char   *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((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 );
 | |
| 
 | |
| 	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, scanbuf, m_ScanParam.Size.dwPhyBytes );
 | |
| 
 | |
| 		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;
 | |
| 
 | |
| 			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;
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 			DBG( _DBG_INFO2, "RedSum   = %lu, ave = %lu\n",
 | |
| 			                            dwSum[0], dwSum[0]/dwPixels );
 | |
| 			DBG( _DBG_INFO2, "GreenSum = %lu, ave = %lu\n",
 | |
| 			                           dwSum[1], dwSum[1] /dwPixels );
 | |
| 			DBG( _DBG_INFO2, "BlueSum  = %lu, ave = %lu\n",
 | |
| 			                           dwSum[2], dwSum[2] /dwPixels );
 | |
| 
 | |
| 			/* do averaging for each channel */
 | |
| 			dwSum[0] /= dwPixels;
 | |
| 			dwSum[1] /= dwPixels;
 | |
| 			dwSum[2] /= dwPixels;
 | |
| 
 | |
| 			adj  = cano_GetNewOffset( dev, dwSum, 0, low, now, high );
 | |
| 			adj |= cano_GetNewOffset( dev, dwSum, 1, low, now, high );
 | |
| 			adj |= cano_GetNewOffset( dev, dwSum, 2, low, now, high );
 | |
| 
 | |
| 			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;
 | |
| 
 | |
| 			for( dw = 0; dw < dwPixels; dw++ )
 | |
| 				dwSum[0] += ((u_short*)scanbuf)[dw];
 | |
| 
 | |
| 			dwSum[0] /= dwPixels;
 | |
| 			DBG( _DBG_INFO2, "Sum=%lu, ave=%lu\n", dwSum[0],dwSum[0]/dwPixels);
 | |
| 
 | |
| 			adj = cano_GetNewOffset( dev, dwSum, 0, low, now, high );
 | |
| 
 | |
| 			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));
 | |
| 	}
 | |
| 
 | |
| 	if( m_ScanParam.bDataType == SCANDATATYPE_Color ) {
 | |
| 		dev->usbDev.a_bRegs[0x38] = now[0];
 | |
| 		dev->usbDev.a_bRegs[0x39] = now[1];
 | |
| 		dev->usbDev.a_bRegs[0x3a] = now[2];
 | |
| 	} else {
 | |
| 		dev->usbDev.a_bRegs[0x38] =
 | |
| 		dev->usbDev.a_bRegs[0x39] =
 | |
| 		dev->usbDev.a_bRegs[0x3a] = now[0];
 | |
| 	}
 | |
| 
 | |
| 	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_char      *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, scanbuf, m_ScanParam.Size.dwTotalBytes );
 | |
| 
 | |
| 		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_char      *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, scanbuf, m_ScanParam.Size.dwTotalBytes );
 | |
| 
 | |
| 		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;
 | |
| 	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.
 | |
| 	 */
 | |
| 	scaps->workaroundFlag &= ~_WAF_SKIP_WHITEFINE;
 | |
| 	scaps->workaroundFlag &= ~_WAF_SKIP_FINE;
 | |
| 	scaps->workaroundFlag &= ~_WAF_BYPASS_CALIBRATION;
 | |
| 
 | |
| 	if( !dev->adj.cacheCalData )
 | |
| 		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;
 | |
| 
 | |
| 		} else if( usb_IsSheetFedDevice(dev)) {
 | |
| 
 | |
| 			/* we only to 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 );
 | |
| 
 | |
| 			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" );
 | |
| 
 | |
| 	return SANE_TRUE;
 | |
| }
 | |
| 
 | |
| /* END PLUSTEK-USBCAL.C .....................................................*/
 |