diff options
author | Jörg Frings-Fürst <debian@jff-webhosting.net> | 2014-10-06 14:00:40 +0200 |
---|---|---|
committer | Jörg Frings-Fürst <debian@jff-webhosting.net> | 2014-10-06 14:00:40 +0200 |
commit | 6e9c41a892ed0e0da326e0278b3221ce3f5713b8 (patch) | |
tree | 2e301d871bbeeb44aa57ff9cc070fcf3be484487 /backend/plustek-usbcalfile.c |
Initial import of sane-backends version 1.0.24-1.2
Diffstat (limited to 'backend/plustek-usbcalfile.c')
-rw-r--r-- | backend/plustek-usbcalfile.c | 845 |
1 files changed, 845 insertions, 0 deletions
diff --git a/backend/plustek-usbcalfile.c b/backend/plustek-usbcalfile.c new file mode 100644 index 0000000..5638b2e --- /dev/null +++ b/backend/plustek-usbcalfile.c @@ -0,0 +1,845 @@ +/*............................................................................. + * Project : SANE library for Plustek flatbed scanners. + *............................................................................. + */ + +/** @file plustek-usbcalfile.c + * @brief Functions for saving/restoring calibration settings + * + * Copyright (C) 2001-2007 Gerhard Jaeger <gerhard@gjaeger.de> + * + * History: + * - 0.46 - first version + * - 0.47 - no changes + * - 0.48 - no changes + * - 0.49 - a_bRegs is now part of the device structure + * - 0.50 - cleanup + * - 0.51 - added functions for saving, reading and restoring + * fine calibration data + * - 0.52 - no changes + * . + * <hr> + * This file is part of the SANE package. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, + * MA 02111-1307, USA. + * + * As a special exception, the authors of SANE give permission for + * additional uses of the libraries contained in this release of SANE. + * + * The exception is that, if you link a SANE library with other files + * to produce an executable, this does not by itself cause the + * resulting executable to be covered by the GNU General Public + * License. Your use of that executable is in no way restricted on + * account of linking the SANE library code into it. + * + * This exception does not, however, invalidate any other reasons why + * the executable file might be covered by the GNU General Public + * License. + * + * If you submit changes to SANE to the maintainers to be included in + * a subsequent release, you agree by submitting the changes that + * those changes may be distributed with this exception intact. + * + * If you write modifications of your own for SANE, it is your choice + * whether to permit this exception to apply to your modifications. + * If you do not wish that, delete this exception notice. + * <hr> + */ + +typedef struct { + u_long red_light_on; + u_long red_light_off; + u_long green_light_on; + u_long green_light_off; + u_long blue_light_on; + u_long blue_light_off; + u_long green_pwm_duty; + +} LightCtrl; + +typedef struct { + u_short version; + + u_short red_gain; + u_short green_gain; + u_short blue_gain; + + u_short red_offs; + u_short green_offs; + u_short blue_offs; + + LightCtrl light; + +} CalData; + +/* our shading buffers */ +static u_short a_wWhiteShading[_SHADING_BUF] = {0}; +static u_short a_wDarkShading[_SHADING_BUF] = {0}; + +/* the version the the calibration files */ +#define _PT_CF_VERSION 0x0002 + +/** function to read a text file and returns the string which starts which + * 'id' string. + * no duplicate entries where detected, always the first occurance will be + * red. + * @param fp - file pointer of file to read + * @param id - what to search for + * @param res - where to store the result upon success + * @return SANE_TRUE on success, SANE_FALSE on any error + */ +static SANE_Bool +usb_ReadSpecLine( FILE *fp, char *id, char* res ) +{ + char tmp[1024]; + char *ptr; + + /* rewind file pointer */ + if( 0 != fseek( fp, 0L, SEEK_SET)) { + DBG( _DBG_ERROR, "fseek: %s\n", strerror(errno)); + return SANE_FALSE; + } + + /* roam through the file and examine each line... */ + while( !feof( fp )) { + + memset( tmp, 0, sizeof(tmp)); + if( NULL != fgets( tmp, 1024, fp )) { + + if( 0 == strncmp( tmp, id, strlen(id))) { + + ptr = &tmp[strlen(id)]; + if( '\0' == *ptr ) + break; + + strcpy( res, ptr ); + res[strlen(res)-1] = '\0'; + return SANE_TRUE; + } + } + } + return SANE_FALSE; +} + +/** function to read data from a file and excluding certain stuff like + * the version lines + * @param fp - file pointer of file to read + * @param except - what to exclude + * @return Pointer to the allocated memory for the data, NULL on any error. + */ +static char* +usb_ReadOtherLines( FILE *fp, char *except ) +{ + char tmp[1024]; + char *ptr, *ptr_base; + int ignore; + int len; + + if( 0 != fseek( fp, 0L, SEEK_END)) + return NULL; + + len = ftell(fp); + + /* rewind file pointer */ + if( 0 != fseek( fp, 0L, SEEK_SET)) + return NULL; + + if( len == 0 ) + return NULL; + + ptr = (char*)malloc(len); + if( NULL == ptr ) + return NULL; + + ptr_base = ptr; + *ptr = '\0'; + ignore = 0; + + /* roam through the file and examine each line... */ + while( !feof( fp )) { + + if( NULL != fgets( tmp, 1024, fp )) { + + /* we ignore the version line... */ + if( 0 == strncmp( tmp, "version=", 8 )) + continue; + + if( !ignore ) { + if(0 != strncmp(tmp, except, strlen(except))) { + + if( strlen( tmp ) > 0 ) { + strcpy( ptr, tmp ); + ptr += strlen(tmp); + *ptr = '\0'; + } + } else { + ignore = 1; + } + } + + /* newline in tmp string resets ignore flag */ + if( strrchr(tmp, '\n')) { + ignore = 0; + } + } + } + return ptr_base; +} + +/** + */ +static SANE_Bool +usb_ReadSamples( FILE *fp, char *which, u_long *dim, u_short *buffer ) +{ + char *p, *next, *rb; + char tmp[1024+30]; + int ignore, diml, c; + u_long val; + + /* rewind file pointer */ + if( 0 != fseek( fp, 0L, SEEK_SET)) + return SANE_FALSE; + + ignore = 0; + diml = 0; + c = 0; + rb = tmp; + *dim = 0; + + /* roam through the file and examine each line... */ + while( !feof( fp )) { + + if( NULL != fgets( rb, 1024, fp )) { + + /* we ignore the version line... */ + if( 0 == strncmp( tmp, "version=", 8 )) + continue; + + p = tmp; + if( !ignore && diml == 0) { + if(0 == strncmp(tmp, which, strlen(which))) { + + /* get dimension */ + diml = strtol(&tmp[strlen(which)], NULL, 10); + p = strchr( &tmp[strlen(which)], ':' ); + p++; + } else { + ignore = 1; + } + } + + /* parse the values... */ + if( !ignore ) { + + rb = tmp; + while( *p ) { + val = strtoul( p, &next, 10 ); + + /* check for error condition */ + if( val == 0 ) { + if( p == next ) { + if( c+1 == diml ) { + *dim = diml; + return SANE_TRUE; + } + break; + } + } + + buffer[c] = (u_short)val; + + /* more values? */ + if( *next == ',') { + p = next+1; + c++; + } else { + p = next; + } + /* reached the end? */ + if( *next == '\0' ) { + + /* we probably have only parsed a part of a value + * so we copy that back to the input buffer and + * parse it the next time... + */ + if( c < diml ) { + sprintf( tmp, "%u", buffer[c] ); + rb = &tmp[strlen(tmp)]; + } + } + } + } + + /* newline in tmp string resets ignore flag */ + if( strrchr(tmp, '\n')) { + ignore = 0; + } + } + } + return SANE_FALSE; +} + +/** + */ +static void +usb_RestoreCalData( Plustek_Device *dev, CalData *cal ) +{ + HWDef *hw = &dev->usbDev.HwSetting; + u_char *regs = dev->usbDev.a_bRegs; + + regs[0x3b] = (u_char)cal->red_gain; + regs[0x3c] = (u_char)cal->green_gain; + regs[0x3d] = (u_char)cal->blue_gain; + regs[0x38] = (u_char)cal->red_offs; + regs[0x39] = (u_char)cal->green_offs; + regs[0x3a] = (u_char)cal->blue_offs; + + regs[0x2a] = _HIBYTE((u_short)cal->light.green_pwm_duty); + regs[0x2b] = _LOBYTE((u_short)cal->light.green_pwm_duty); + + regs[0x2c] = _HIBYTE((u_short)cal->light.red_light_on); + regs[0x2d] = _LOBYTE((u_short)cal->light.red_light_on); + regs[0x2e] = _HIBYTE((u_short)cal->light.red_light_off); + regs[0x2f] = _LOBYTE((u_short)cal->light.red_light_off); + + regs[0x30] = _HIBYTE((u_short)cal->light.green_light_on); + regs[0x31] = _LOBYTE((u_short)cal->light.green_light_on); + regs[0x32] = _HIBYTE((u_short)cal->light.green_light_off); + regs[0x33] = _LOBYTE((u_short)cal->light.green_light_off); + + regs[0x34] = _HIBYTE((u_short)cal->light.blue_light_on); + regs[0x35] = _LOBYTE((u_short)cal->light.blue_light_on); + regs[0x36] = _HIBYTE((u_short)cal->light.blue_light_off); + regs[0x37] = _LOBYTE((u_short)cal->light.blue_light_off); + + hw->red_lamp_on = (u_short)cal->light.red_light_on; + hw->red_lamp_off = (u_short)cal->light.red_light_off; + hw->green_lamp_on = (u_short)cal->light.green_light_on; + hw->green_lamp_off = (u_short)cal->light.green_light_off; + hw->blue_lamp_on = (u_short)cal->light.blue_light_on; + hw->blue_lamp_off = (u_short)cal->light.blue_light_off; +} + +/** + */ +static void +usb_CreatePrefix( Plustek_Device *dev, char *pfx, SANE_Bool add_bitdepth ) +{ + char bd[5]; + ScanDef *scanning = &dev->scanning; + ScanParam *param = &scanning->sParam; + + switch( scanning->sParam.bSource ) { + + case SOURCE_Transparency: strcpy( pfx, "tpa-" ); break; + case SOURCE_Negative: strcpy( pfx, "neg-" ); break; + case SOURCE_ADF: strcpy( pfx, "adf-" ); break; + default: pfx[0] = '\0'; break; + } + + sprintf( bd, "%u=", param->bBitDepth ); + if( param->bDataType == SCANDATATYPE_Color ) + strcat( pfx, "color" ); + else + strcat( pfx, "gray" ); + + if( add_bitdepth ) + strcat( pfx, bd ); +} + +/** function to read and set the calibration data from external file + */ +static SANE_Bool +usb_ReadAndSetCalData( Plustek_Device *dev ) +{ + char pfx[20]; + char tmp[1024]; + u_short version; + int res; + FILE *fp; + CalData cal; + SANE_Bool ret; + + DBG( _DBG_INFO, "usb_ReadAndSetCalData()\n" ); + + if( usb_InCalibrationMode(dev)) { + DBG( _DBG_INFO, "- we are in calibration mode!\n" ); + return SANE_FALSE; + } + + if( NULL == dev->calFile ) { + DBG( _DBG_ERROR, "- No calibration filename set!\n" ); + return SANE_FALSE; + } + + sprintf( tmp, "%s-coarse.cal", dev->calFile ); + DBG( _DBG_INFO, "- Reading coarse calibration data from file\n"); + DBG( _DBG_INFO, " %s\n", tmp ); + + fp = fopen( tmp, "r" ); + if( NULL == fp ) { + DBG( _DBG_ERROR, "File %s not found\n", tmp ); + return SANE_FALSE; + } + + /* check version */ + if( !usb_ReadSpecLine( fp, "version=", tmp )) { + DBG( _DBG_ERROR, "Could not find version info!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + DBG( _DBG_INFO, "- Calibration file version: %s\n", tmp ); + if( 1 != sscanf( tmp, "0x%04hx", &version )) { + DBG( _DBG_ERROR, "Could not decode version info!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + if( version != _PT_CF_VERSION ) { + DBG( _DBG_ERROR, "Versions do not match!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + usb_CreatePrefix( dev, pfx, SANE_TRUE ); + + ret = SANE_FALSE; + if( usb_ReadSpecLine( fp, pfx, tmp )) { + DBG( _DBG_INFO, "- Calibration data: %s\n", tmp ); + + res = sscanf( tmp, "%hu,%hu,%hu,%hu,%hu,%hu," + "%lu,%lu,%lu,%lu,%lu,%lu,%lu\n", + &cal.red_gain, &cal.red_offs, + &cal.green_gain, &cal.green_offs, + &cal.blue_gain, &cal.blue_offs, + &cal.light.red_light_on, &cal.light.red_light_off, + &cal.light.green_light_on, &cal.light.green_light_off, + &cal.light.blue_light_on, &cal.light.blue_light_off, + &cal.light.green_pwm_duty ); + + if( 13 == res ) { + usb_RestoreCalData( dev, &cal ); + ret = SANE_TRUE; + } else { + DBG( _DBG_ERROR, "Error reading coarse-calibration data, only " + "%d elements available!\n", res ); + } + } else { + DBG( _DBG_ERROR, "Error reading coarse-calibration data for " + "PFX: >%s<!\n", pfx ); + } + + fclose( fp ); + DBG( _DBG_INFO, "usb_ReadAndSetCalData() done -> %u\n", ret ); + + return ret; +} + +/** + */ +static void +usb_PrepCalData( Plustek_Device *dev, CalData *cal ) +{ + u_char *regs = dev->usbDev.a_bRegs; + + memset( cal, 0, sizeof(CalData)); + cal->version = _PT_CF_VERSION; + + cal->red_gain = (u_short)regs[0x3b]; + cal->green_gain = (u_short)regs[0x3c]; + cal->blue_gain = (u_short)regs[0x3d]; + cal->red_offs = (u_short)regs[0x38]; + cal->green_offs = (u_short)regs[0x39]; + cal->blue_offs = (u_short)regs[0x3a]; + + cal->light.green_pwm_duty = regs[0x2a] * 256 + regs[0x2b]; + + cal->light.red_light_on = regs[0x2c] * 256 + regs[0x2d]; + cal->light.red_light_off = regs[0x2e] * 256 + regs[0x2f]; + cal->light.green_light_on = regs[0x30] * 256 + regs[0x31]; + cal->light.green_light_off = regs[0x32] * 256 + regs[0x33]; + cal->light.blue_light_on = regs[0x34] * 256 + regs[0x35]; + cal->light.blue_light_off = regs[0x36] * 256 + regs[0x37]; +} + +/** function to save/update the calibration data + */ +static void +usb_SaveCalData( Plustek_Device *dev ) +{ + char pfx[20]; + char fn[1024]; + char tmp[1024]; + char set_tmp[1024]; + char *other_tmp; + u_short version; + FILE *fp; + CalData cal; + ScanDef *scanning = &dev->scanning; + + DBG( _DBG_INFO, "usb_SaveCalData()\n" ); + + /* no new data, so skip this step too */ + if( SANE_TRUE == scanning->skipCoarseCalib ) { + DBG( _DBG_INFO, "- No calibration data to save!\n" ); + return; + } + + if( NULL == dev->calFile ) { + DBG( _DBG_ERROR, "- No calibration filename set!\n" ); + return; + } + + sprintf( fn, "%s-coarse.cal", dev->calFile ); + DBG( _DBG_INFO, "- Saving coarse calibration data to file\n" ); + DBG( _DBG_INFO, " %s\n", fn ); + + usb_PrepCalData ( dev, &cal ); + usb_CreatePrefix( dev, pfx, SANE_TRUE ); + DBG( _DBG_INFO2, "- PFX: >%s<\n", pfx ); + + sprintf( set_tmp, "%s%u,%u,%u,%u,%u,%u," + "%lu,%lu,%lu,%lu,%lu,%lu,%lu\n", pfx, + cal.red_gain, cal.red_offs, + cal.green_gain, cal.green_offs, + cal.blue_gain, cal.blue_offs, + cal.light.red_light_on, cal.light.red_light_off, + cal.light.green_light_on, cal.light.green_light_off, + cal.light.blue_light_on, cal.light.blue_light_off, + cal.light.green_pwm_duty ); + + /* read complete old file if compatible... */ + other_tmp = NULL; + fp = fopen( fn, "r+" ); + if( NULL != fp ) { + + if( usb_ReadSpecLine( fp, "version=", tmp )) { + DBG( _DBG_INFO, "- Calibration file version: %s\n", tmp ); + + if( 1 == sscanf( tmp, "0x%04hx", &version )) { + + if( version == cal.version ) { + + DBG( _DBG_INFO, "- Versions do match\n" ); + + /* read the rest... */ + other_tmp = usb_ReadOtherLines( fp, pfx ); + } else { + DBG( _DBG_INFO2, "- Versions do not match (0x%04x)\n", version ); + } + } else { + DBG( _DBG_INFO2, "- cannot decode version\n" ); + } + } else { + DBG( _DBG_INFO2, "- Version not found\n" ); + } + fclose( fp ); + } + fp = fopen( fn, "w+" ); + if( NULL == fp ) { + DBG( _DBG_ERROR, "- Cannot create file %s\n", fn ); + DBG( _DBG_ERROR, "- -> %s\n", strerror(errno)); + if( other_tmp ) + free( other_tmp ); + return; + } + + /* rewrite the file again... */ + fprintf( fp, "version=0x%04X\n", cal.version ); + if( strlen( set_tmp )) + fprintf( fp, "%s", set_tmp ); + + if( other_tmp ) { + fprintf( fp, "%s", other_tmp ); + free( other_tmp ); + } + fclose( fp ); + DBG( _DBG_INFO, "usb_SaveCalData() done.\n" ); +} + +/** + */ +static void +usb_SaveFineCalData( Plustek_Device *dev, int dpi, + u_short *dark, u_short *white, u_long vals ) +{ + char pfx[30]; + char fn[1024]; + char tmp[1024]; + char *other_tmp; + u_short version; + u_long i; + FILE *fp; + + if( NULL == dev->calFile ) { + DBG( _DBG_ERROR, "- No calibration filename set!\n" ); + return; + } + + sprintf( fn, "%s-fine.cal", dev->calFile ); + DBG( _DBG_INFO, "- Saving fine calibration data to file\n" ); + DBG( _DBG_INFO, " %s\n", fn ); + + usb_CreatePrefix( dev, pfx, SANE_FALSE ); + sprintf( tmp, "%s:%u", pfx, dpi ); + strcpy( pfx, tmp ); + DBG( _DBG_INFO2, "- PFX: >%s<\n", pfx ); + + /* read complete old file if compatible... */ + other_tmp = NULL; + fp = fopen( fn, "r+" ); + if( NULL != fp ) { + + if( usb_ReadSpecLine( fp, "version=", tmp )) { + DBG( _DBG_INFO, "- Calibration file version: %s\n", tmp ); + + if( 1 == sscanf( tmp, "0x%04hx", &version )) { + + if( version == _PT_CF_VERSION ) { + DBG( _DBG_INFO, "- Versions do match\n" ); + + /* read the rest... */ + other_tmp = usb_ReadOtherLines( fp, pfx ); + } else { + DBG( _DBG_INFO2, "- Versions do not match (0x%04x)\n", version ); + } + } else { + DBG( _DBG_INFO2, "- cannot decode version\n" ); + } + } else { + DBG( _DBG_INFO2, "- Version not found\n" ); + } + fclose( fp ); + } + + fp = fopen( fn, "w+" ); + if( NULL == fp ) { + DBG( _DBG_ERROR, "- Cannot create file %s\n", fn ); + return; + } + + /* rewrite the file again... */ + fprintf( fp, "version=0x%04X\n", _PT_CF_VERSION ); + + if( other_tmp ) { + fprintf( fp, "%s", other_tmp ); + free( other_tmp ); + } + + fprintf( fp, "%s:dark:dim=%lu:", pfx, vals ); + for( i=0; i<vals-1; i++ ) + fprintf( fp, "%u,", dark[i]); + fprintf( fp, "%u\n", dark[vals-1]); + + fprintf( fp, "%s:white:dim=%lu:", pfx, vals ); + for( i=0; i<vals-1; i++ ) + fprintf( fp, "%u,", white[i]); + fprintf( fp, "%u\n", white[vals-1]); + + fclose( fp ); +} + +/** function to read and set the calibration data from external file + */ +static SANE_Bool +usb_ReadFineCalData( Plustek_Device *dev, int dpi, + u_long *dim_d, u_short *dark, + u_long *dim_w, u_short *white ) +{ + char pfx[30]; + char tmp[1024]; + u_short version; + FILE *fp; + + DBG( _DBG_INFO, "usb_ReadFineCalData()\n" ); + if( usb_InCalibrationMode(dev)) { + DBG( _DBG_INFO, "- we are in calibration mode!\n" ); + return SANE_FALSE; + } + + if( NULL == dev->calFile ) { + DBG( _DBG_ERROR, "- No calibration filename set!\n" ); + return SANE_FALSE; + } + + sprintf( tmp, "%s-fine.cal", dev->calFile ); + DBG( _DBG_INFO, "- Reading fine calibration data from file\n"); + DBG( _DBG_INFO, " %s\n", tmp ); + + *dim_d = *dim_w = 0; + + fp = fopen( tmp, "r" ); + if( NULL == fp ) { + DBG( _DBG_ERROR, "File %s not found\n", tmp ); + return SANE_FALSE; + } + + /* check version */ + if( !usb_ReadSpecLine( fp, "version=", tmp )) { + DBG( _DBG_ERROR, "Could not find version info!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + DBG( _DBG_INFO, "- Calibration file version: %s\n", tmp ); + if( 1 != sscanf( tmp, "0x%04hx", &version )) { + DBG( _DBG_ERROR, "Could not decode version info!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + if( version != _PT_CF_VERSION ) { + DBG( _DBG_ERROR, "Versions do not match!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + usb_CreatePrefix( dev, pfx, SANE_FALSE ); + + sprintf( tmp, "%s:%u:%s:dim=", pfx, dpi, "dark" ); + if( !usb_ReadSamples( fp, tmp, dim_d, dark )) { + DBG( _DBG_ERROR, "Error reading dark-calibration data!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + sprintf( tmp, "%s:%u:%s:dim=", pfx, dpi, "white" ); + if( !usb_ReadSamples( fp, tmp, dim_w, white )) { + DBG( _DBG_ERROR, "Error reading white-calibration data!\n" ); + fclose( fp ); + return SANE_FALSE; + } + + fclose( fp ); + return SANE_TRUE; +} + +/** + */ +static void +usb_get_shading_part(u_short *buf, u_long offs, u_long src_len, int dst_len) +{ + u_short *p_src, *p_dst; + int i, j; + + if (src_len == 0 || dst_len == 0) + return; + + p_dst = buf; + for (i=0; i<3; i++) { + + p_src = buf + src_len * i + offs; + + for (j=0; j<dst_len; j++) { + + *(p_dst++) = *(p_src++); + } + } +} + +/** function to read the fine calibration results from file + * and to set the correct part of the calibration buffers for + * storing in the device + * @param dev - the almigthy device structure + * @returns SANE_FALSE when the reading fails or SANE_TRUE on success + */ +static SANE_Bool +usb_FineShadingFromFile( Plustek_Device *dev ) +{ + ScanDef *scan = &dev->scanning; + ScanParam *sp = &scan->sParam; + u_short xdpi; + u_long dim_w, dim_d, offs; + + xdpi = usb_SetAsicDpiX( dev, sp->UserDpi.x ); + + if( !usb_ReadFineCalData( dev, xdpi, &dim_d, a_wDarkShading, + &dim_w, a_wWhiteShading)) { + return SANE_FALSE; + } + + /* now we need to get the correct part of the line... */ + dim_d /= 3; + dim_w /= 3; + + offs = ((u_long)sp->Origin.x * xdpi) / 300; + + usb_GetPhyPixels( dev, sp ); + + DBG( _DBG_INFO2, "FINE Calibration from file:\n" ); + DBG( _DBG_INFO2, "XDPI = %u\n", xdpi ); + DBG( _DBG_INFO2, "Dim = %lu\n", dim_d ); + DBG( _DBG_INFO2, "Pixels = %lu\n", sp->Size.dwPixels ); + DBG( _DBG_INFO2, "PhyPixels = %lu\n", sp->Size.dwPhyPixels ); + DBG( _DBG_INFO2, "Origin.X = %u\n", sp->Origin.x ); + DBG( _DBG_INFO2, "Offset = %lu\n", offs ); + + usb_get_shading_part(a_wDarkShading, offs, dim_d, sp->Size.dwPhyPixels); + usb_get_shading_part(a_wWhiteShading, offs, dim_w, sp->Size.dwPhyPixels); + + return SANE_TRUE; +} + +/** function to save the fine calibration results and to set the correct part + * of the calibration buffers for storing in the device + * @param dev - the almigthy device structure + * @param tmp_sp - intermediate scan parameter + */ +static void +usb_SaveCalSetShading( Plustek_Device *dev, ScanParam *tmp_sp ) +{ + ScanParam *sp = &dev->scanning.sParam; + u_short xdpi; + u_long offs; + + if( !dev->adj.cacheCalData ) + return; + + /* save the values */ + xdpi = usb_SetAsicDpiX( dev, tmp_sp->UserDpi.x ); + + usb_SaveFineCalData( dev, xdpi, a_wDarkShading, + a_wWhiteShading, tmp_sp->Size.dwPixels*3 ); + + /* now we need to get the correct part of the line... */ + xdpi = usb_SetAsicDpiX( dev, sp->UserDpi.x ); + offs = ((u_long)sp->Origin.x * xdpi) / 300; + usb_GetPhyPixels( dev, sp ); + + DBG( _DBG_INFO2, "FINE Calibration area after saving:\n" ); + DBG( _DBG_INFO2, "XDPI = %u\n", xdpi ); + DBG( _DBG_INFO2, "Dim = %lu\n", tmp_sp->Size.dwPixels ); + DBG( _DBG_INFO2, "Pixels = %lu\n", sp->Size.dwPixels ); + DBG( _DBG_INFO2, "PhyPixels = %lu\n", sp->Size.dwPhyPixels ); + DBG( _DBG_INFO2, "Origin.X = %u\n", sp->Origin.x ); + DBG( _DBG_INFO2, "Offset = %lu\n", offs ); + + if (!usb_InCalibrationMode(dev)) { + + usb_get_shading_part( a_wDarkShading, offs, + tmp_sp->Size.dwPixels, sp->Size.dwPhyPixels ); + usb_get_shading_part( a_wWhiteShading, offs, + tmp_sp->Size.dwPixels, sp->Size.dwPhyPixels ); + + memcpy( tmp_sp, sp, sizeof(ScanParam)); + tmp_sp->bBitDepth = 16; + + usb_GetPhyPixels( dev, tmp_sp ); + } +} + +/* END PLUSTEK-USBCALFILE.C .................................................*/ |