/* HP Scanjet 3900 series - SANE Backend controller Copyright (C) 2005-2009 Jonathan Bravo Lopez <jkdsoft@gmail.com> 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. */ /* Backend Code for SANE*/ #define HP3900_CONFIG_FILE "hp3900.conf" #define GAMMA_DEFAULT 1.0 #include "../include/sane/config.h" #include "../include/sane/sane.h" #include "../include/sane/sanei.h" #include "../include/sane/sanei_backend.h" #include "../include/sane/sanei_config.h" #include "../include/sane/saneopts.h" #include "../include/sane/sanei_usb.h" #include "../include/sane/sanei_debug.h" #include "hp3900_rts8822.c" struct st_convert { SANE_Int colormode; SANE_Int depth; SANE_Int threshold; SANE_Int negative; SANE_Int real_depth; }; /* options enumerator */ typedef enum { opt_begin = 0, grp_geometry, opt_tlx, opt_tly, opt_brx, opt_bry, opt_resolution, /* gamma tables */ opt_gamma_red, opt_gamma_green, opt_gamma_blue, opt_scantype, opt_colormode, opt_depth, opt_threshold, /* debugging options */ grp_debug, opt_model, opt_negative, opt_nogamma, opt_nowshading, opt_realdepth, opt_emulategray, opt_nowarmup, opt_dbgimages, opt_reset, /* device information */ grp_info, opt_chipname, opt_chipid, opt_scancount, opt_infoupdate, /* supported buttons. RTS8822 supports up to 6 buttons */ grp_sensors, opt_button_0, opt_button_1, opt_button_2, opt_button_3, opt_button_4, opt_button_5, opt_count } EOptionIndex; /* linked list of SANE_Device structures */ typedef struct TDevListEntry { struct TDevListEntry *pNext; SANE_Device dev; char *devname; } TDevListEntry; typedef struct { char *pszVendor; char *pszName; } TScannerModel; typedef union { SANE_Word w; SANE_Word *wa; /* word array */ SANE_String s; } TOptionValue; typedef struct { SANE_Int model; SANE_Option_Descriptor aOptions[opt_count]; TOptionValue aValues[opt_count]; struct params ScanParams; /* lists */ SANE_String_Const *list_colormodes; SANE_Int *list_depths; SANE_String_Const *list_models; SANE_Int *list_resolutions; SANE_String_Const *list_sources; SANE_Word *aGammaTable[3]; /* a 16-to-16 bit color lookup table */ SANE_Range rng_gamma; /* reading image */ SANE_Byte *image; SANE_Byte *rest; SANE_Int rest_amount; SANE_Int mylin; /* convertion settings */ struct st_convert cnv; /* ranges */ SANE_Range rng_threshold; SANE_Range rng_horizontal; SANE_Range rng_vertical; SANE_Int scan_count; SANE_Int fScanning; /* TRUE if actively scanning */ } TScanner; /* functions to manage backend's options */ static void options_init (TScanner * scanner); static void options_free (TScanner * scanner); /* devices listing */ static SANE_Int _ReportDevice (TScannerModel * pModel, const char *pszDeviceName); static SANE_Status attach_one_device (SANE_String_Const devname); /* capabilities */ static SANE_Status bknd_colormodes (TScanner * scanner, SANE_Int model); static void bknd_constrains (TScanner * scanner, SANE_Int source, SANE_Int type); static SANE_Status bknd_depths (TScanner * scanner, SANE_Int model); static SANE_Status bknd_info (TScanner * scanner); static SANE_Status bknd_models (TScanner * scanner); static SANE_Status bknd_resolutions (TScanner * scanner, SANE_Int model); static SANE_Status bknd_sources (TScanner * scanner, SANE_Int model); /* convertions */ static void Color_Negative (SANE_Byte * buffer, SANE_Int size, SANE_Int depth); static void Color_to_Gray (SANE_Byte * buffer, SANE_Int size, SANE_Int depth); static void Gray_to_Lineart (SANE_Byte * buffer, SANE_Int size, SANE_Int threshold); static void Depth_16_to_8 (SANE_Byte * from_buffer, SANE_Int size, SANE_Byte * to_buffer); /* gamma functions */ static void gamma_apply (TScanner * s, SANE_Byte * buffer, SANE_Int size, SANE_Int depth); static SANE_Int gamma_create (TScanner * s, double gamma); static void gamma_free (TScanner * s); static SANE_Int Get_Colormode (SANE_String colormode); static SANE_Int Get_Model (SANE_String model); static SANE_Int Get_Source (SANE_String source); static SANE_Int GetUSB_device_model (SANE_String_Const name); static size_t max_string_size (const SANE_String_Const strings[]); static SANE_Status get_button_status (TScanner * s); /* reading buffers */ static SANE_Status img_buffers_alloc (TScanner * scanner, SANE_Int size); static SANE_Status img_buffers_free (TScanner * scanner); static SANE_Status option_get (TScanner * scanner, SANE_Int optid, void *result); static SANE_Status option_set (TScanner * scanner, SANE_Int optid, void *value, SANE_Int * pInfo); static void Set_Coordinates (SANE_Int scantype, SANE_Int resolution, struct st_coords *coords); static SANE_Int set_ScannerModel (SANE_Int proposed, SANE_Int product, SANE_Int vendor); static void Silent_Compile (void); static SANE_Status Translate_coords (struct st_coords *coords); /* SANE functions */ void sane_cancel (SANE_Handle h); void sane_close (SANE_Handle h); SANE_Status sane_control_option (SANE_Handle h, SANE_Int n, SANE_Action Action, void *pVal, SANE_Int * pInfo); void sane_exit (void); SANE_Status sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only); const SANE_Option_Descriptor *sane_get_option_descriptor (SANE_Handle h, SANE_Int n); SANE_Status sane_get_parameters (SANE_Handle h, SANE_Parameters * p); SANE_Status sane_get_select_fd (SANE_Handle handle, SANE_Int * fd); SANE_Status sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize); SANE_Status sane_open (SANE_String_Const name, SANE_Handle * h); SANE_Status sane_read (SANE_Handle h, SANE_Byte * buf, SANE_Int maxlen, SANE_Int * len); SANE_Status sane_set_io_mode (SANE_Handle handle, SANE_Bool non_blocking); SANE_Status sane_start (SANE_Handle h); /* variables */ static struct st_device *device = NULL; static TDevListEntry *_pFirstSaneDev = 0; static SANE_Int iNumSaneDev = 0; static const SANE_Device **_pSaneDevList = 0; /* Own functions */ static SANE_Status bknd_resolutions (TScanner * scanner, SANE_Int model) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_resolutions(*scanner, model=%i)\n", model); if (scanner != NULL) { SANE_Int *res = NULL; switch (model) { case BQ5550: case UA4900: { SANE_Int myres[] = { 8, 50, 75, 100, 150, 200, 300, 600, 1200 }; res = (SANE_Int *) malloc (sizeof (myres)); if (res != NULL) memcpy (res, &myres, sizeof (myres)); } break; case HPG2710: case HP3800: { /* 1200 and 2400 dpi are disabled until problems are solved */ SANE_Int myres[] = { 7, 50, 75, 100, 150, 200, 300, 600 }; res = (SANE_Int *) malloc (sizeof (myres)); if (res != NULL) memcpy (res, &myres, sizeof (myres)); } break; case HP4370: case HPG3010: case HPG3110: { SANE_Int myres[] = { 10, 50, 75, 100, 150, 200, 300, 600, 1200, 2400, 4800 }; res = (SANE_Int *) malloc (sizeof (myres)); if (res != NULL) memcpy (res, &myres, sizeof (myres)); } break; default: /* HP3970 & HP4070 & UA4900 */ { SANE_Int myres[] = { 9, 50, 75, 100, 150, 200, 300, 600, 1200, 2400 }; res = (SANE_Int *) malloc (sizeof (myres)); if (res != NULL) memcpy (res, &myres, sizeof (myres)); } break; } if (res != NULL) { if (scanner->list_resolutions != NULL) free (scanner->list_resolutions); scanner->list_resolutions = res; rst = SANE_STATUS_GOOD; } } return rst; } static SANE_Status bknd_models (TScanner * scanner) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_models:\n"); if (scanner != NULL) { SANE_String_Const *model = NULL; /* at this moment all devices use the same list */ SANE_String_Const mymodel[] = { "HP3800", "HP3970", "HP4070", "HP4370", "UA4900", "HPG3010", "BQ5550", "HPG2710", "HPG3110", 0 }; /* allocate space to save list */ model = (SANE_String_Const *) malloc (sizeof (mymodel)); if (model != NULL) memcpy (model, &mymodel, sizeof (mymodel)); if (model != NULL) { /* free previous list */ if (scanner->list_models != NULL) free (scanner->list_models); /* set new list */ scanner->list_models = model; rst = SANE_STATUS_GOOD; } } return rst; } static SANE_Status bknd_colormodes (TScanner * scanner, SANE_Int model) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_colormodes(*scanner, model=%i)\n", model); if (scanner != NULL) { SANE_String_Const *colormode = NULL; /* at this moment all devices use the same list */ SANE_String_Const mycolormode[] = { SANE_VALUE_SCAN_MODE_COLOR, SANE_VALUE_SCAN_MODE_GRAY, SANE_VALUE_SCAN_MODE_LINEART, 0 }; /* silence gcc */ model = model; colormode = (SANE_String_Const *) malloc (sizeof (mycolormode)); if (colormode != NULL) memcpy (colormode, &mycolormode, sizeof (mycolormode)); if (colormode != NULL) { if (scanner->list_colormodes != NULL) free (scanner->list_colormodes); scanner->list_colormodes = colormode; rst = SANE_STATUS_GOOD; } } return rst; } static SANE_Status bknd_sources (TScanner * scanner, SANE_Int model) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_sources(*scanner, model=%i)\n", model); if (scanner != NULL) { SANE_String_Const *source = NULL; switch (model) { case UA4900: { SANE_String_Const mysource[] = { SANE_I18N ("Flatbed"), 0 }; source = (SANE_String_Const *) malloc (sizeof (mysource)); if (source != NULL) memcpy (source, &mysource, sizeof (mysource)); } break; default: /* hp3970, hp4070, hp4370 and others */ { SANE_String_Const mysource[] = { SANE_I18N ("Flatbed"), SANE_I18N ("Slide"), SANE_I18N ("Negative"), 0 }; source = (SANE_String_Const *) malloc (sizeof (mysource)); if (source != NULL) memcpy (source, &mysource, sizeof (mysource)); } break; } if (source != NULL) { if (scanner->list_sources != NULL) free (scanner->list_sources); scanner->list_sources = source; rst = SANE_STATUS_GOOD; } } return rst; } static SANE_Status bknd_depths (TScanner * scanner, SANE_Int model) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_depths(*scanner, model=%i\n", model); if (scanner != NULL) { SANE_Int *depth = NULL; /* at this moment all devices use the same list */ SANE_Int mydepth[] = { 2, 8, 16 }; /*{3, 8, 12, 16}; */ /* silence gcc */ model = model; depth = (SANE_Int *) malloc (sizeof (mydepth)); if (depth != NULL) memcpy (depth, &mydepth, sizeof (mydepth)); if (depth != NULL) { if (scanner->list_depths != NULL) free (scanner->list_depths); scanner->list_depths = depth; rst = SANE_STATUS_GOOD; } } return rst; } static SANE_Status bknd_info (TScanner * scanner) { SANE_Status rst = SANE_STATUS_INVAL; DBG (DBG_FNC, "> bknd_info(*scanner)"); if (scanner != NULL) { char data[256]; /* update chipset name */ Chipset_Name (device, data, 255); if (scanner->aValues[opt_chipname].s != NULL) { free (scanner->aValues[opt_chipname].s); scanner->aValues[opt_chipname].s = NULL; } scanner->aValues[opt_chipname].s = strdup (data); scanner->aOptions[opt_chipname].size = strlen (data) + 1; /* update chipset id */ scanner->aValues[opt_chipid].w = Chipset_ID (device); /* update scans counter */ scanner->aValues[opt_scancount].w = RTS_ScanCounter_Get (device); rst = SANE_STATUS_GOOD; } return rst; } static SANE_Int GetUSB_device_model (SANE_String_Const name) { SANE_Int usbid, model; /* default model is unknown */ model = -1; /* open usb device */ if (sanei_usb_open (name, &usbid) == SANE_STATUS_GOOD) { SANE_Int vendor, product; if (sanei_usb_get_vendor_product (usbid, &vendor, &product) == SANE_STATUS_GOOD) model = Device_get (product, vendor); sanei_usb_close (usbid); } return model; } static void Silent_Compile (void) { /* There are some functions in hp3900_rts8822.c that aren't used yet. To avoid compilation warnings we will use them here */ SANE_Byte a = 1; if (a == 0) { Buttons_Status (device); Calib_WriteTable (device, NULL, 0, 0); Gamma_GetTables (device, NULL); } } static void bknd_constrains (TScanner * scanner, SANE_Int source, SANE_Int type) { struct st_coords *coords = Constrains_Get (device, source); if ((coords != NULL) && (scanner != NULL)) { switch (type) { case 1: /* Y */ scanner->rng_vertical.max = coords->height; break; default: /* X */ scanner->rng_horizontal.max = coords->width; break; } } } static SANE_Status img_buffers_free (TScanner * scanner) { if (scanner != NULL) { if (scanner->image != NULL) { free (scanner->image); scanner->image = NULL; } if (scanner->rest != NULL) { free (scanner->rest); scanner->rest = NULL; } scanner->rest_amount = 0; } return SANE_STATUS_GOOD; } static SANE_Status img_buffers_alloc (TScanner * scanner, SANE_Int size) { SANE_Status rst; /* default result at this point */ rst = SANE_STATUS_INVAL; if (scanner != NULL) { /* default result at this point */ rst = SANE_STATUS_NO_MEM; /* free previous allocs */ img_buffers_free (scanner); scanner->image = (SANE_Byte *) malloc (size * sizeof (SANE_Byte)); if (scanner->image != NULL) { scanner->rest = (SANE_Byte *) malloc (size * sizeof (SANE_Byte)); if (scanner->rest != NULL) rst = SANE_STATUS_GOOD; /* ok !! */ } if (rst != SANE_STATUS_GOOD) img_buffers_free (scanner); } return rst; } static SANE_Int set_ScannerModel (SANE_Int proposed, SANE_Int product, SANE_Int vendor) { /* This function will set the device behaviour */ SANE_Int current = Device_get (product, vendor); char *sdevname[10] = { "Unknown", "HP3970", "HP4070", "HP4370", "UA4900", "HP3800", "HPG3010", "BQ5550", "HPG2710", "HPG3110" }; DBG (DBG_FNC, "> set_ScannerModel(proposed=%i, product=%04x, vendor=%04x)\n", proposed, product, vendor); if (proposed < 0) { if ((current < 0) || (current >= DEVSCOUNT)) { DBG (DBG_VRB, " -> Unknown device. Defaulting to HP3970...\n"); RTS_Debug->dev_model = HP3970; } else { RTS_Debug->dev_model = current; DBG (DBG_VRB, " -> Device model is %s\n", sdevname[current + 1]); } } else { if (proposed < DEVSCOUNT) { RTS_Debug->dev_model = proposed; DBG (DBG_VRB, " -> Device %s , treating as %s ...\n", sdevname[current + 1], sdevname[proposed + 1]); } else { if ((current >= 0) && (current < DEVSCOUNT)) { RTS_Debug->dev_model = current; DBG (DBG_VRB, " -> Device not supported. Defaulting to %s ...\n", sdevname[current + 1]); } else { RTS_Debug->dev_model = HP3970; DBG (DBG_VRB, "-> Device not supported. Defaulting to HP3970...\n"); } } } return OK; } static void Set_Coordinates (SANE_Int scantype, SANE_Int resolution, struct st_coords *coords) { struct st_coords *limits = Constrains_Get (device, scantype); DBG (DBG_FNC, "> Set_Coordinates(res=%i, *coords):\n", resolution); if (coords->left == -1) coords->left = 0; if (coords->width == -1) coords->width = limits->width; if (coords->top == -1) coords->top = 0; if (coords->height == -1) coords->height = limits->height; DBG (DBG_FNC, " -> Coords [MM] : xy(%i, %i) wh(%i, %i)\n", coords->left, coords->top, coords->width, coords->height); coords->left = MM_TO_PIXEL (coords->left, resolution); coords->width = MM_TO_PIXEL (coords->width, resolution); coords->top = MM_TO_PIXEL (coords->top, resolution); coords->height = MM_TO_PIXEL (coords->height, resolution); DBG (DBG_FNC, " -> Coords [px] : xy(%i, %i) wh(%i, %i)\n", coords->left, coords->top, coords->width, coords->height); Constrains_Check (device, resolution, scantype, coords); DBG (DBG_FNC, " -> Coords [check]: xy(%i, %i) wh(%i, %i)\n", coords->left, coords->top, coords->width, coords->height); } static void Color_Negative (SANE_Byte * buffer, SANE_Int size, SANE_Int depth) { if (buffer != NULL) { SANE_Int a; SANE_Int max_value = (1 << depth) - 1; if (depth > 8) { USHORT *sColor = (void *) buffer; for (a = 0; a < size / 2; a++) { *sColor = max_value - *sColor; sColor++; } } else { for (a = 0; a < size; a++) *(buffer + a) = max_value - *(buffer + a); } } } static SANE_Status get_button_status (TScanner * s) { if (s != NULL) { SANE_Int a, b, status, btn; b = 1; status = Buttons_Released (device) & 63; for (a = 0; a < 6; a++) { if ((status & b) != 0) { btn = Buttons_Order (device, b); if (btn != -1) s->aValues[opt_button_0 + btn].w = SANE_TRUE; } b <<= 1; } } return SANE_STATUS_GOOD; } static void Depth_16_to_8 (SANE_Byte * from_buffer, SANE_Int size, SANE_Byte * to_buffer) { if ((from_buffer != NULL) && (to_buffer != NULL)) { SANE_Int a, b; a = 1; b = 0; while (a < size) { *(to_buffer + b) = *(from_buffer + a); a += 2; b++; } } } static void Gray_to_Lineart (SANE_Byte * buffer, SANE_Int size, SANE_Int threshold) { /* code provided by tobias leutwein */ if (buffer != NULL) { SANE_Byte toBufferByte; SANE_Int fromBufferPos_i = 0; SANE_Int toBufferPos_i = 0; SANE_Int bitPos_i; while (fromBufferPos_i < size) { toBufferByte = 0; for (bitPos_i = 7; bitPos_i != (-1); bitPos_i--) { if ((fromBufferPos_i < size) && (buffer[fromBufferPos_i] < threshold)) toBufferByte |= (1u << bitPos_i); fromBufferPos_i++; } buffer[toBufferPos_i] = toBufferByte; toBufferPos_i++; } } } static void Color_to_Gray (SANE_Byte * buffer, SANE_Int size, SANE_Int depth) { /* converts 3 color channel into 1 gray channel of specified bit depth */ if (buffer != NULL) { SANE_Int c, chn, chn_size; SANE_Byte *ptr_src = NULL; SANE_Byte *ptr_dst = NULL; float data, chn_data; float coef[3] = { 0.299, 0.587, 0.114 }; /* coefficients per channel */ chn_size = (depth > 8) ? 2 : 1; ptr_src = (void *) buffer; ptr_dst = (void *) buffer; for (c = 0; c < size / (3 * chn_size); c++) { data = 0.; /* get, apply coeffs and sum channels */ for (chn = 0; chn < 3; chn++) { chn_data = data_lsb_get (ptr_src + (chn * chn_size), chn_size); data += (chn_data * coef[chn]); } /* save result */ data_lsb_set (ptr_dst, (SANE_Int) data, chn_size); ptr_src += 3 * chn_size; /* next triplet */ ptr_dst += chn_size; } } } static void gamma_free (TScanner * s) { DBG (DBG_FNC, "> gamma_free()\n"); if (s != NULL) { /* Destroy gamma tables */ SANE_Int a; for (a = CL_RED; a <= CL_BLUE; a++) { if (s->aGammaTable[a] != NULL) { free (s->aGammaTable[a]); s->aGammaTable[a] = NULL; } } } } static SANE_Int gamma_create (TScanner * s, double gamma) { SANE_Int rst = ERROR; /* by default */ DBG (DBG_FNC, "> gamma_create(*s)\n"); if (s != NULL) { SANE_Int a; double value, c; /* default result */ rst = OK; /* destroy previus gamma tables */ gamma_free (s); /* check gamma value */ if (gamma < 0) gamma = GAMMA_DEFAULT; /* allocate space for 16 bit gamma tables */ for (a = CL_RED; a <= CL_BLUE; a++) { s->aGammaTable[a] = malloc (65536 * sizeof (SANE_Word)); if (s->aGammaTable[a] == NULL) { rst = ERROR; break; } } if (rst == OK) { /* fill tables */ for (a = 0; a < 65536; a++) { value = (a / (65536. - 1)); value = pow (value, (1. / gamma)); value = value * (65536. - 1); c = (SANE_Int) value; if (c > (65536. - 1)) c = (65536. - 1); else if (c < 0) c = 0; s->aGammaTable[CL_RED][a] = c; s->aGammaTable[CL_GREEN][a] = c; s->aGammaTable[CL_BLUE][a] = c; } } else gamma_free (s); } return rst; } static void gamma_apply (TScanner * s, SANE_Byte * buffer, SANE_Int size, SANE_Int depth) { if ((s != NULL) && (buffer != NULL)) { SANE_Int c; SANE_Int dot_size = 3 * ((depth > 8) ? 2 : 1); SANE_Byte *pColor = buffer; USHORT *sColor = (void *) buffer; if ((s->aGammaTable[CL_RED] != NULL) && (s->aGammaTable[CL_GREEN] != NULL) && (s->aGammaTable[CL_BLUE] != NULL)) { for (c = 0; c < size / dot_size; c++) { if (depth > 8) { *sColor = s->aGammaTable[CL_RED][*sColor]; *(sColor + 1) = s->aGammaTable[CL_GREEN][*(sColor + 1)]; *(sColor + 2) = s->aGammaTable[CL_BLUE][*(sColor + 2)]; sColor += 3; } else { /* 8 bits gamma */ *pColor = (s->aGammaTable[CL_RED][*pColor * 256] >> 8) & 0xff; *(pColor + 1) = (s-> aGammaTable[CL_GREEN][*(pColor + 1) * 256] >> 8) & 0xff; *(pColor + 2) = (s-> aGammaTable[CL_BLUE][*(pColor + 2) * 256] >> 8) & 0xff; pColor += 3; } } } } } static SANE_Int Get_Model (SANE_String model) { SANE_Int rst; if (strcmp (model, "HP3800") == 0) rst = HP3800; else if (strcmp (model, "HPG2710") == 0) rst = HPG2710; else if (strcmp (model, "HP3970") == 0) rst = HP3970; else if (strcmp (model, "HP4070") == 0) rst = HP4070; else if (strcmp (model, "HP4370") == 0) rst = HP4370; else if (strcmp (model, "HPG3010") == 0) rst = HPG3010; else if (strcmp (model, "HPG3110") == 0) rst = HPG3110; else if (strcmp (model, "UA4900") == 0) rst = UA4900; else if (strcmp (model, "BQ5550") == 0) rst = BQ5550; else rst = HP3970; /* default */ return rst; } static SANE_Int Get_Source (SANE_String source) { SANE_Int rst; if (strcmp (source, SANE_I18N ("Flatbed")) == 0) rst = ST_NORMAL; else if (strcmp (source, SANE_I18N ("Slide")) == 0) rst = ST_TA; else if (strcmp (source, SANE_I18N ("Negative")) == 0) rst = ST_NEG; else rst = ST_NORMAL; /* default */ return rst; } static SANE_Int Get_Colormode (SANE_String colormode) { SANE_Int rst; if (strcmp (colormode, SANE_VALUE_SCAN_MODE_COLOR) == 0) rst = CM_COLOR; else if (strcmp (colormode, SANE_VALUE_SCAN_MODE_GRAY) == 0) rst = CM_GRAY; else if (strcmp (colormode, SANE_VALUE_SCAN_MODE_LINEART) == 0) rst = CM_LINEART; else rst = CM_COLOR; /* default */ return rst; } static SANE_Status Translate_coords (struct st_coords *coords) { SANE_Int data; DBG (DBG_FNC, "> Translate_coords(*coords)\n"); if ((coords->left < 0) || (coords->top < 0) || (coords->width < 0) || (coords->height < 0)) return SANE_STATUS_INVAL; if (coords->width < coords->left) { data = coords->left; coords->left = coords->width; coords->width = data; } if (coords->height < coords->top) { data = coords->top; coords->top = coords->height; coords->height = data; } coords->width -= coords->left; coords->height -= coords->top; if (coords->width == 0) coords->width++; if (coords->height == 0) coords->height++; return SANE_STATUS_GOOD; } static size_t max_string_size (const SANE_String_Const strings[]) { size_t size, max_size = 0; SANE_Int i; DBG (DBG_FNC, "> max_string_size:\n"); for (i = 0; strings[i]; ++i) { size = strlen (strings[i]) + 1; if (size > max_size) max_size = size; } return max_size; } static void options_free (TScanner * scanner) { /* frees all information contained in controls */ DBG (DBG_FNC, "> options_free\n"); if (scanner != NULL) { SANE_Int i; SANE_Option_Descriptor *pDesc; TOptionValue *pVal; /* free gamma tables */ gamma_free (scanner); /* free lists */ if (scanner->list_resolutions != NULL) free (scanner->list_resolutions); if (scanner->list_depths != NULL) free (scanner->list_depths); if (scanner->list_sources != NULL) free (scanner->list_sources); if (scanner->list_colormodes != NULL) free (scanner->list_colormodes); if (scanner->list_models != NULL) free (scanner->list_models); /* free values in certain controls */ for (i = opt_begin; i < opt_count; i++) { pDesc = &scanner->aOptions[i]; pVal = &scanner->aValues[i]; if (pDesc->type == SANE_TYPE_STRING) { if (pVal->s != NULL) free (pVal->s); } } } } static void options_init (TScanner * scanner) { /* initializes all controls */ DBG (DBG_FNC, "> options_init\n"); if (scanner != NULL) { SANE_Int i; SANE_Option_Descriptor *pDesc; TOptionValue *pVal; /* set gamma */ gamma_create (scanner, 1.0); /* color convertion */ scanner->cnv.colormode = -1; scanner->cnv.negative = FALSE; scanner->cnv.threshold = 40; scanner->cnv.real_depth = FALSE; scanner->cnv.depth = -1; /* setting threshold */ scanner->rng_threshold.min = 0; scanner->rng_threshold.max = 255; scanner->rng_threshold.quant = 0; /* setting gamma range (16 bits depth) */ scanner->rng_gamma.min = 0; scanner->rng_gamma.max = 65535; scanner->rng_gamma.quant = 0; /* setting default horizontal constrain in milimeters */ scanner->rng_horizontal.min = 0; scanner->rng_horizontal.max = 220; scanner->rng_horizontal.quant = 1; /* setting default vertical constrain in milimeters */ scanner->rng_vertical.min = 0; scanner->rng_vertical.max = 300; scanner->rng_vertical.quant = 1; /* allocate option lists */ bknd_info (scanner); bknd_colormodes (scanner, RTS_Debug->dev_model); bknd_depths (scanner, RTS_Debug->dev_model); bknd_models (scanner); bknd_resolutions (scanner, RTS_Debug->dev_model); bknd_sources (scanner, RTS_Debug->dev_model); /* By default preview scan */ scanner->ScanParams.scantype = ST_NORMAL; scanner->ScanParams.colormode = CM_COLOR; scanner->ScanParams.resolution_x = 75; scanner->ScanParams.resolution_y = 75; scanner->ScanParams.coords.left = 0; scanner->ScanParams.coords.top = 0; scanner->ScanParams.coords.width = 220; scanner->ScanParams.coords.height = 300; scanner->ScanParams.depth = 8; scanner->ScanParams.channel = 0; for (i = opt_begin; i < opt_count; i++) { pDesc = &scanner->aOptions[i]; pVal = &scanner->aValues[i]; /* defaults */ pDesc->name = ""; pDesc->title = ""; pDesc->desc = ""; pDesc->type = SANE_TYPE_INT; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->cap = 0; switch (i) { case opt_begin: pDesc->title = SANE_TITLE_NUM_OPTIONS; pDesc->desc = SANE_DESC_NUM_OPTIONS; pDesc->cap = SANE_CAP_SOFT_DETECT; pVal->w = (SANE_Word) opt_count; break; case grp_geometry: pDesc->name = SANE_NAME_GEOMETRY; pDesc->title = SANE_TITLE_GEOMETRY; pDesc->desc = SANE_DESC_GEOMETRY; pDesc->type = SANE_TYPE_GROUP; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->cap = 0; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pVal->w = 0; break; case opt_tlx: pDesc->name = SANE_NAME_SCAN_TL_X; pDesc->title = SANE_TITLE_SCAN_TL_X; pDesc->desc = SANE_DESC_SCAN_TL_X; pDesc->unit = SANE_UNIT_MM; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_horizontal; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = 0; break; case opt_tly: pDesc->name = SANE_NAME_SCAN_TL_Y; pDesc->title = SANE_TITLE_SCAN_TL_Y; pDesc->desc = SANE_DESC_SCAN_TL_Y; pDesc->unit = SANE_UNIT_MM; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_vertical; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = 0; break; case opt_brx: pDesc->name = SANE_NAME_SCAN_BR_X; pDesc->title = SANE_TITLE_SCAN_BR_X; pDesc->desc = SANE_DESC_SCAN_BR_X; pDesc->unit = SANE_UNIT_MM; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_horizontal; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = scanner->rng_horizontal.max; break; case opt_bry: pDesc->name = SANE_NAME_SCAN_BR_Y; pDesc->title = SANE_TITLE_SCAN_BR_Y; pDesc->desc = SANE_DESC_SCAN_BR_Y; pDesc->unit = SANE_UNIT_MM; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_vertical; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = scanner->rng_vertical.max; break; case opt_resolution: pDesc->name = SANE_NAME_SCAN_RESOLUTION; pDesc->title = SANE_TITLE_SCAN_RESOLUTION; pDesc->desc = SANE_DESC_SCAN_RESOLUTION; pDesc->unit = SANE_UNIT_DPI; pDesc->constraint_type = SANE_CONSTRAINT_WORD_LIST; pDesc->constraint.word_list = scanner->list_resolutions; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = scanner->list_resolutions[1]; break; case opt_gamma_red: pDesc->name = SANE_NAME_GAMMA_VECTOR_R; pDesc->title = SANE_TITLE_GAMMA_VECTOR_R; pDesc->desc = SANE_DESC_GAMMA_VECTOR_R; pDesc->size = scanner->rng_gamma.max * sizeof (SANE_Word); pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_gamma; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->wa = scanner->aGammaTable[CL_RED]; break; case opt_gamma_green: pDesc->name = SANE_NAME_GAMMA_VECTOR_G; pDesc->title = SANE_TITLE_GAMMA_VECTOR_G; pDesc->desc = SANE_DESC_GAMMA_VECTOR_G; pDesc->size = scanner->rng_gamma.max * sizeof (SANE_Word); pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_gamma; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->wa = scanner->aGammaTable[CL_GREEN]; break; case opt_gamma_blue: pDesc->name = SANE_NAME_GAMMA_VECTOR_B; pDesc->title = SANE_TITLE_GAMMA_VECTOR_B; pDesc->desc = SANE_DESC_GAMMA_VECTOR_B; pDesc->size = scanner->rng_gamma.max * sizeof (SANE_Word); pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_gamma; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->wa = scanner->aGammaTable[CL_BLUE]; break; case opt_scantype: pDesc->name = SANE_NAME_SCAN_SOURCE; pDesc->title = SANE_TITLE_SCAN_SOURCE; pDesc->desc = SANE_DESC_SCAN_SOURCE; pDesc->type = SANE_TYPE_STRING; pDesc->size = max_string_size (scanner->list_sources); pDesc->constraint_type = SANE_CONSTRAINT_STRING_LIST; pDesc->constraint.string_list = scanner->list_sources; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->s = strdup (scanner->list_sources[0]); break; case opt_colormode: pDesc->name = SANE_NAME_SCAN_MODE; pDesc->title = SANE_TITLE_SCAN_MODE; pDesc->desc = SANE_DESC_SCAN_MODE; pDesc->type = SANE_TYPE_STRING; pDesc->size = max_string_size (scanner->list_colormodes); pDesc->constraint_type = SANE_CONSTRAINT_STRING_LIST; pDesc->constraint.string_list = scanner->list_colormodes; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->s = strdup (scanner->list_colormodes[0]); break; case opt_depth: pDesc->name = SANE_NAME_BIT_DEPTH; pDesc->title = SANE_TITLE_BIT_DEPTH; pDesc->desc = SANE_DESC_BIT_DEPTH; pDesc->type = SANE_TYPE_INT; pDesc->unit = SANE_UNIT_BIT; pDesc->constraint_type = SANE_CONSTRAINT_WORD_LIST; pDesc->constraint.word_list = scanner->list_depths; pDesc->cap = SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->w = scanner->list_depths[1]; break; case opt_threshold: pDesc->name = SANE_NAME_THRESHOLD; pDesc->title = SANE_TITLE_THRESHOLD; pDesc->desc = SANE_DESC_THRESHOLD; pDesc->type = SANE_TYPE_INT; pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_RANGE; pDesc->constraint.range = &scanner->rng_threshold; pDesc->cap |= SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT | SANE_CAP_INACTIVE; pVal->w = 0x80; break; /* debugging options */ case grp_debug: pDesc->name = "grp_debug"; pDesc->title = SANE_I18N ("Debugging Options"); pDesc->desc = ""; pDesc->type = SANE_TYPE_GROUP; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->cap = SANE_CAP_ADVANCED; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pVal->w = 0; break; case opt_model: pDesc->name = "opt_model"; pDesc->title = SANE_I18N ("Scanner model"); pDesc->desc = SANE_I18N ("Allows one to test device behaviour with other supported models"); pDesc->type = SANE_TYPE_STRING; pDesc->size = max_string_size (scanner->list_models); pDesc->constraint_type = SANE_CONSTRAINT_STRING_LIST; pDesc->constraint.string_list = scanner->list_models; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_SELECT | SANE_CAP_SOFT_DETECT; pVal->s = strdup (scanner->list_models[0]); break; case opt_negative: pDesc->name = "opt_negative"; pDesc->title = SANE_I18N ("Negative"); pDesc->desc = SANE_I18N ("Image colours will be inverted"); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_nogamma: pDesc->name = "opt_nogamma"; pDesc->title = SANE_I18N ("Disable gamma correction"); pDesc->desc = SANE_I18N ("Gamma correction will be disabled"); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_nowshading: pDesc->name = "opt_nowshading"; pDesc->title = SANE_I18N ("Disable white shading correction"); pDesc->desc = SANE_I18N ("White shading correction will be disabled"); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_nowarmup: pDesc->name = "opt_nowarmup"; pDesc->title = SANE_I18N ("Skip warmup process"); pDesc->desc = SANE_I18N ("Warmup process will be disabled"); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_realdepth: pDesc->name = "opt_realdepth"; pDesc->title = SANE_I18N ("Force real depth"); pDesc->desc = SANE_I18N ("If gamma is enabled, scans are always made in 16 bits depth to improve image quality and then converted to the selected depth. This option avoids depth emulation."); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_emulategray: pDesc->name = "opt_emulategray"; pDesc->title = SANE_I18N ("Emulate Grayscale"); pDesc->desc = SANE_I18N ("If enabled, image will be scanned in color mode and then converted to grayscale by software. This may improve image quality in some circumstances."); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_dbgimages: pDesc->name = "opt_dbgimages"; pDesc->title = SANE_I18N ("Save debugging images"); pDesc->desc = SANE_I18N ("If enabled, some images involved in scanner processing are saved to analyze them."); pDesc->type = SANE_TYPE_BOOL; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT | SANE_CAP_SOFT_SELECT; pVal->w = SANE_FALSE; break; case opt_reset: pDesc->name = "opt_reset"; pDesc->title = SANE_I18N ("Reset chipset"); pDesc->desc = SANE_I18N ("Resets chipset data"); pDesc->type = SANE_TYPE_BUTTON; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.string_list = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_SELECT; pVal->w = 0; break; /* device information */ case grp_info: pDesc->name = "grp_info"; pDesc->title = SANE_I18N ("Information"); pDesc->desc = ""; pDesc->type = SANE_TYPE_GROUP; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->cap = 0; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pVal->w = 0; break; case opt_chipname: pDesc->name = "opt_chipname"; pDesc->title = SANE_I18N ("Chipset name"); pDesc->desc = SANE_I18N ("Shows chipset name used in device."); pDesc->type = SANE_TYPE_STRING; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT; pVal->s = strdup (SANE_I18N ("Unknown")); pDesc->size = strlen(pVal->s) + 1; break; case opt_chipid: pDesc->name = "opt_chipid"; pDesc->title = SANE_I18N ("Chipset ID"); pDesc->desc = SANE_I18N ("Shows the chipset ID"); pDesc->type = SANE_TYPE_INT; pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT; pVal->w = -1; break; case opt_scancount: pDesc->name = "opt_scancount"; pDesc->title = SANE_I18N ("Scan counter"); pDesc->desc = SANE_I18N ("Shows the number of scans made by scanner"); pDesc->type = SANE_TYPE_INT; pDesc->unit = SANE_UNIT_NONE; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_DETECT; pVal->w = -1; break; case opt_infoupdate: pDesc->name = "opt_infoupdate"; pDesc->title = SANE_I18N ("Update information"); pDesc->desc = SANE_I18N ("Updates information about device"); pDesc->type = SANE_TYPE_BUTTON; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.string_list = 0; pDesc->cap = SANE_CAP_ADVANCED | SANE_CAP_SOFT_SELECT; pVal->w = 0; break; /* buttons support */ case grp_sensors: pDesc->name = SANE_NAME_SENSORS; pDesc->title = SANE_TITLE_SENSORS; pDesc->desc = SANE_DESC_SENSORS; pDesc->type = SANE_TYPE_GROUP; pDesc->unit = SANE_UNIT_NONE; pDesc->size = 0; pDesc->cap = 0; pDesc->constraint_type = SANE_CONSTRAINT_NONE; pDesc->constraint.range = 0; pVal->w = 0; break; case opt_button_0: case opt_button_1: case opt_button_2: case opt_button_3: case opt_button_4: case opt_button_5: { char name[12]; char title[128]; sprintf (name, "button %d", i - opt_button_0); sprintf (title, "Scanner button %d", i - opt_button_0); pDesc->name = strdup (name); pDesc->title = strdup (title); pDesc->desc = SANE_I18N ("This option reflects a front panel scanner button"); pDesc->type = SANE_TYPE_BOOL; pDesc->cap = SANE_CAP_SOFT_DETECT | SANE_CAP_ADVANCED; if (i - opt_button_0 >= Buttons_Count (device)) pDesc->cap |= SANE_CAP_INACTIVE; pDesc->unit = SANE_UNIT_NONE; pDesc->size = sizeof (SANE_Word); pDesc->constraint_type = SANE_CONSTRAINT_NONE; pVal->w = SANE_FALSE; } break; } } } } static SANE_Int _ReportDevice (TScannerModel * pModel, const char *pszDeviceName) { SANE_Int rst = ERROR; TDevListEntry *pNew, *pDev; DBG (DBG_FNC, "> _ReportDevice:\n"); pNew = malloc (sizeof (TDevListEntry)); if (pNew != NULL) { rst = OK; /* add new element to the end of the list */ if (_pFirstSaneDev != NULL) { /* Add at the end of existing list */ for (pDev = _pFirstSaneDev; pDev->pNext; pDev = pDev->pNext); pDev->pNext = pNew; } else _pFirstSaneDev = pNew; /* fill in new element */ pNew->pNext = NULL; pNew->devname = (char *) strdup (pszDeviceName); pNew->dev.name = pNew->devname; pNew->dev.vendor = pModel->pszVendor; pNew->dev.model = pModel->pszName; pNew->dev.type = SANE_I18N ("flatbed scanner"); iNumSaneDev++; } return rst; } static SANE_Status attach_one_device (SANE_String_Const devname) { static TScannerModel sModel; DBG (DBG_FNC, "> attach_one_device(devname=%s)\n", devname); switch (GetUSB_device_model (devname)) { case HP3800: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet 3800"); break; case HPG2710: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet G2710"); break; case HP3970: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet 3970"); break; case HP4070: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet 4070 Photosmart"); break; case HP4370: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet 4370"); break; case HPG3010: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet G3010"); break; case HPG3110: sModel.pszVendor = (char *) strdup ("Hewlett-Packard"); sModel.pszName = (char *) strdup ("Scanjet G3110"); break; case UA4900: sModel.pszVendor = (char *) strdup ("UMAX"); sModel.pszName = (char *) strdup ("Astra 4900"); break; case BQ5550: sModel.pszVendor = (char *) strdup ("BenQ"); sModel.pszName = (char *) strdup ("5550"); break; default: sModel.pszVendor = (char *) strdup ("Unknown"); sModel.pszName = (char *) strdup ("RTS8822 chipset based"); break; } _ReportDevice (&sModel, devname); return SANE_STATUS_GOOD; } /* Sane default functions */ SANE_Status sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize) { FILE *conf_fp; /* Config file stream */ SANE_Char line[PATH_MAX]; SANE_Char *str = NULL; SANE_String_Const proper_str; SANE_Int nline = 0; /* Initialize debug */ DBG_INIT (); DBG (DBG_FNC, "> sane_init\n"); /* silence gcc */ authorize = authorize; /* Initialize usb */ sanei_usb_init (); /* Parse config file */ conf_fp = sanei_config_open (HP3900_CONFIG_FILE); if (conf_fp) { while (sanei_config_read (line, sizeof (line), conf_fp)) { nline++; if (str) free (str); proper_str = sanei_config_get_string (line, &str); /* Discards white lines and comments */ if ((str != NULL) && (proper_str != line) && (str[0] != '#')) { /* If line's not blank or a comment, then it's the device * filename or a usb directive. */ sanei_usb_attach_matching_devices (line, attach_one_device); } } fclose (conf_fp); } else { /* default */ DBG (DBG_VRB, "- %s not found. Looking for hardcoded usb ids ...\n", HP3900_CONFIG_FILE); sanei_usb_attach_matching_devices ("usb 0x03f0 0x2605", attach_one_device); /* HP3800 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x2805", attach_one_device); /* HPG2710 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x2305", attach_one_device); /* HP3970 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x2405", attach_one_device); /* HP4070 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x4105", attach_one_device); /* HP4370 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x4205", attach_one_device); /* HPG3010 */ sanei_usb_attach_matching_devices ("usb 0x03f0 0x4305", attach_one_device); /* HPG3110 */ sanei_usb_attach_matching_devices ("usb 0x06dc 0x0020", attach_one_device); /* UA4900 */ sanei_usb_attach_matching_devices ("usb 0x04a5 0x2211", attach_one_device); /* BQ5550 */ } /* Return backend version */ if (version_code != NULL) *version_code = SANE_VERSION_CODE (SANE_CURRENT_MAJOR, V_MINOR, 0); return SANE_STATUS_GOOD; } SANE_Status sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only) { SANE_Status rst = SANE_STATUS_GOOD; local_only = local_only; if (_pSaneDevList) free (_pSaneDevList); _pSaneDevList = malloc (sizeof (*_pSaneDevList) * (iNumSaneDev + 1)); if (_pSaneDevList != NULL) { TDevListEntry *pDev; SANE_Int i = 0; for (pDev = _pFirstSaneDev; pDev; pDev = pDev->pNext) _pSaneDevList[i++] = &pDev->dev; _pSaneDevList[i++] = 0; /* last entry is 0 */ *device_list = _pSaneDevList; } else rst = SANE_STATUS_NO_MEM; DBG (DBG_FNC, "> sane_get_devices: %i\n", rst); return rst; } SANE_Status sane_open (SANE_String_Const name, SANE_Handle * h) { TScanner *s; SANE_Status rst; /* check the name */ if (strlen (name) == 0) /* default to first available device */ name = _pFirstSaneDev->dev.name; /* allocate space for RTS environment */ device = RTS_Alloc (); if (device != NULL) { /* Open device */ rst = sanei_usb_open (name, &device->usb_handle); if (rst == SANE_STATUS_GOOD) { /* Allocating memory for device */ s = malloc (sizeof (TScanner)); if (s != NULL) { memset (s, 0, sizeof (TScanner)); /* Initializing RTS */ if (Init_Vars () == OK) { SANE_Int vendor, product; /* Setting device model */ if (sanei_usb_get_vendor_product (device->usb_handle, &vendor, &product) == SANE_STATUS_GOOD) s->model = Device_get (product, vendor); else s->model = HP3970; set_ScannerModel (s->model, product, vendor); /* Initialize device */ if (RTS_Scanner_Init (device) == OK) { /* silencing unused functions */ Silent_Compile (); /* initialize backend options */ options_init (s); *h = s; /* everything went ok */ rst = SANE_STATUS_GOOD; } else { free ((void *) s); rst = SANE_STATUS_INVAL; } } else rst = SANE_STATUS_NO_MEM; } else rst = SANE_STATUS_NO_MEM; } } else rst = SANE_STATUS_NO_MEM; DBG (DBG_FNC, "> sane_open(name=%s): %i\n", name, rst); return rst; } const SANE_Option_Descriptor * sane_get_option_descriptor (SANE_Handle h, SANE_Int n) { SANE_Option_Descriptor *rst = NULL; if ((n >= opt_begin) && (n < opt_count)) { TScanner *s = (TScanner *) h; rst = &s->aOptions[n]; } DBG (DBG_FNC, "> SANE_Option_Descriptor(handle, n=%i): %i\n", n, (rst == NULL) ? -1 : 0); return rst; } static SANE_Status option_get (TScanner * scanner, SANE_Int optid, void *result) { /* This function returns value contained in selected option */ DBG (DBG_FNC, "> option_get(optid=%i)\n", optid); if ((scanner != NULL) && (result != NULL)) { switch (optid) { /* SANE_Word */ case opt_begin: /* null */ case opt_reset: /* null */ case opt_negative: case opt_nogamma: case opt_nowshading: case opt_emulategray: case opt_dbgimages: case opt_nowarmup: case opt_realdepth: case opt_depth: case opt_resolution: case opt_threshold: case opt_brx: case opt_tlx: case opt_bry: case opt_tly: *(SANE_Word *) result = scanner->aValues[optid].w; break; /* SANE_Int */ case opt_chipid: case opt_scancount: *(SANE_Int *) result = scanner->aValues[optid].w; break; /* SANE_Word array */ case opt_gamma_red: case opt_gamma_green: case opt_gamma_blue: memcpy (result, scanner->aValues[optid].wa, scanner->aOptions[optid].size); break; /* String */ case opt_colormode: case opt_scantype: case opt_model: case opt_chipname: strncpy (result, scanner->aValues[optid].s, scanner->aOptions[optid].size); ((char*)result)[scanner->aOptions[optid].size-1] = '\0'; break; /* scanner buttons */ case opt_button_0: get_button_status (scanner); case opt_button_1: case opt_button_2: case opt_button_3: case opt_button_4: case opt_button_5: /* copy the button state */ *(SANE_Word *) result = scanner->aValues[optid].w; /* clear the button state */ scanner->aValues[optid].w = SANE_FALSE; break; } } return SANE_STATUS_GOOD; } static SANE_Status option_set (TScanner * scanner, SANE_Int optid, void *value, SANE_Int * pInfo) { SANE_Status rst; DBG (DBG_FNC, "> option_set(optid=%i)\n", optid); rst = SANE_STATUS_INVAL; if (scanner != NULL) { if (scanner->fScanning == FALSE) { SANE_Int info = 0; rst = SANE_STATUS_GOOD; switch (optid) { case opt_brx: case opt_tlx: case opt_bry: case opt_tly: case opt_depth: case opt_nogamma: case opt_nowshading: case opt_nowarmup: case opt_negative: case opt_emulategray: case opt_dbgimages: case opt_threshold: case opt_resolution: info |= SANE_INFO_RELOAD_PARAMS; scanner->aValues[optid].w = *(SANE_Word *) value; break; case opt_gamma_red: case opt_gamma_green: case opt_gamma_blue: memcpy (scanner->aValues[optid].wa, value, scanner->aOptions[optid].size); break; case opt_scantype: if (strcmp (scanner->aValues[optid].s, value) != 0) { struct st_coords *coords; SANE_Int source; if (scanner->aValues[optid].s) free (scanner->aValues[optid].s); scanner->aValues[optid].s = strdup (value); source = Get_Source (scanner->aValues[opt_scantype].s); coords = Constrains_Get (device, source); if (coords != NULL) { bknd_constrains (scanner, source, 0); bknd_constrains (scanner, source, 1); scanner->aValues[opt_tlx].w = 0; scanner->aValues[opt_tly].w = 0; scanner->aValues[opt_brx].w = coords->width; scanner->aValues[opt_bry].w = coords->height; } info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; } break; case opt_colormode: if (strcmp (scanner->aValues[optid].s, value) != 0) { if (scanner->aValues[optid].s) free (scanner->aValues[optid].s); scanner->aValues[optid].s = strdup (value); if (Get_Colormode (scanner->aValues[optid].s) == CM_LINEART) scanner->aOptions[opt_threshold].cap &= ~SANE_CAP_INACTIVE; else scanner->aOptions[opt_threshold].cap |= SANE_CAP_INACTIVE; info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; } break; case opt_model: if (strcmp (scanner->aValues[optid].s, value) != 0) { SANE_Int model; if (scanner->aValues[optid].s) free (scanner->aValues[optid].s); scanner->aValues[optid].s = strdup (value); model = Get_Model (scanner->aValues[optid].s); if (model != RTS_Debug->dev_model) { SANE_Int source; struct st_coords *coords; /* free configuration of last model */ Free_Config (device); /* set new model */ RTS_Debug->dev_model = model; /* and load configuration of current model */ Load_Config (device); /* update options according to selected device */ bknd_info (scanner); bknd_colormodes (scanner, model); bknd_depths (scanner, model); bknd_resolutions (scanner, model); bknd_sources (scanner, model); /* updating lists */ scanner->aOptions[opt_colormode].size = max_string_size (scanner->list_colormodes); scanner->aOptions[opt_colormode].constraint. string_list = scanner->list_colormodes; scanner->aOptions[opt_depth].constraint.word_list = scanner->list_depths; scanner->aOptions[opt_resolution].constraint.word_list = scanner->list_resolutions; scanner->aOptions[opt_scantype].size = max_string_size (scanner->list_sources); scanner->aOptions[opt_scantype].constraint.string_list = scanner->list_sources; /* default values */ if (scanner->aValues[opt_colormode].s != NULL) free (scanner->aValues[opt_colormode].s); if (scanner->aValues[opt_scantype].s != NULL) free (scanner->aValues[opt_scantype].s); scanner->aValues[opt_colormode].s = strdup (scanner->list_colormodes[0]); scanner->aValues[opt_scantype].s = strdup (scanner->list_sources[0]); scanner->aValues[opt_resolution].w = scanner->list_resolutions[1]; scanner->aValues[opt_depth].w = scanner->list_depths[1]; source = Get_Source (scanner->aValues[opt_scantype].s); coords = Constrains_Get (device, source); if (coords != NULL) { bknd_constrains (scanner, source, 0); bknd_constrains (scanner, source, 1); scanner->aValues[opt_tlx].w = 0; scanner->aValues[opt_tly].w = 0; scanner->aValues[opt_brx].w = coords->width; scanner->aValues[opt_bry].w = coords->height; } } info |= SANE_INFO_RELOAD_PARAMS | SANE_INFO_RELOAD_OPTIONS; } break; case opt_reset: Chipset_Reset (device); break; case opt_realdepth: scanner->aValues[optid].w = (scanner->cnv.real_depth == TRUE) ? SANE_TRUE : SANE_FALSE; break; case opt_infoupdate: if (bknd_info (scanner) == SANE_STATUS_GOOD) info |= SANE_INFO_RELOAD_OPTIONS; break; default: rst = SANE_STATUS_INVAL; break; } if (pInfo != NULL) *pInfo = info; } } return rst; } SANE_Status sane_control_option (SANE_Handle h, SANE_Int n, SANE_Action Action, void *pVal, SANE_Int * pInfo) { TScanner *scanner; SANE_Status rst; DBG (DBG_FNC, "> sane_control_option\n"); scanner = (TScanner *) h; switch (Action) { case SANE_ACTION_GET_VALUE: rst = option_get (scanner, n, pVal); break; case SANE_ACTION_SET_VALUE: rst = option_set (scanner, n, pVal, pInfo); break; case SANE_ACTION_SET_AUTO: rst = SANE_STATUS_UNSUPPORTED; break; default: rst = SANE_STATUS_INVAL; break; } return rst; } SANE_Status sane_get_parameters (SANE_Handle h, SANE_Parameters * p) { SANE_Status rst = SANE_STATUS_INVAL; TScanner *s = (TScanner *) h; DBG (DBG_FNC, "+ sane_get_parameters:"); if (s != NULL) { struct st_coords coords; SANE_Int res, source, depth, colormode, frameformat, bpl; /* first do some checks */ /* colormode */ colormode = Get_Colormode (s->aValues[opt_colormode].s); /* frameformat */ frameformat = (colormode == CM_COLOR) ? SANE_FRAME_RGB : SANE_FRAME_GRAY; /* depth */ depth = (colormode == CM_LINEART) ? 1 : s->aValues[opt_depth].w; /* scan type */ source = Get_Source (s->aValues[opt_scantype].s); /* resolution */ res = s->aValues[opt_resolution].w; /* image coordinates in milimeters */ coords.left = s->aValues[opt_tlx].w; coords.top = s->aValues[opt_tly].w; coords.width = s->aValues[opt_brx].w; coords.height = s->aValues[opt_bry].w; /* validate coords */ if (Translate_coords (&coords) == SANE_STATUS_GOOD) { Set_Coordinates (source, res, &coords); if (colormode != CM_LINEART) { bpl = coords.width * ((depth > 8) ? 2 : 1); if (colormode == CM_COLOR) bpl *= 3; /* three channels */ } else bpl = (coords.width + 7) / 8; /* return the data */ p->format = frameformat; p->last_frame = SANE_TRUE; p->depth = depth; p->lines = coords.height; p->pixels_per_line = coords.width; p->bytes_per_line = bpl; DBG (DBG_FNC, " -> Depth : %i\n", depth); DBG (DBG_FNC, " -> Height: %i\n", coords.height); DBG (DBG_FNC, " -> Width : %i\n", coords.width); DBG (DBG_FNC, " -> BPL : %i\n", bpl); rst = SANE_STATUS_GOOD; } } DBG (DBG_FNC, "- sane_get_parameters: %i\n", rst); return rst; } SANE_Status sane_start (SANE_Handle h) { SANE_Status rst = SANE_STATUS_INVAL; TScanner *s; DBG (DBG_FNC, "+ sane_start\n"); s = (TScanner *) h; if (s != NULL) { struct st_coords coords; SANE_Int res, source, colormode, depth, channel; /* first do some checks */ /* Get Scan type */ source = Get_Source (s->aValues[opt_scantype].s); /* Check if scanner supports slides and negatives in case selected source is tma */ if (!((source != ST_NORMAL) && (RTS_isTmaAttached (device) == FALSE))) { /* Get depth */ depth = s->aValues[opt_depth].w; /* Get color mode */ colormode = Get_Colormode (s->aValues[opt_colormode].s); /* Emulating certain color modes */ if (colormode == CM_LINEART) { /* emulate lineart */ s->cnv.colormode = CM_LINEART; colormode = CM_GRAY; depth = 8; } else if ((colormode == CM_GRAY) && (s->aValues[opt_emulategray].w == SANE_TRUE)) { /* emulate grayscale */ s->cnv.colormode = CM_GRAY; colormode = CM_COLOR; } else s->cnv.colormode = -1; /* setting channel for colormodes different than CM_COLOR */ channel = (colormode != CM_COLOR) ? 1 : 0; /* negative colors */ s->cnv.negative = (s->aValues[opt_negative].w == SANE_TRUE) ? TRUE : FALSE; /* Get threshold */ s->cnv.threshold = s->aValues[opt_threshold].w; /* Get resolution */ res = s->aValues[opt_resolution].w; /* set depth emulation */ if (s->cnv.colormode == CM_LINEART) s->cnv.real_depth = TRUE; else s->cnv.real_depth = (s->aValues[opt_realdepth].w == SANE_TRUE) ? TRUE : FALSE; /* use gamma? */ RTS_Debug->EnableGamma = (s->aValues[opt_nogamma].w == SANE_TRUE) ? FALSE : TRUE; /* disable white shading correction? */ RTS_Debug->wshading = (s->aValues[opt_nowshading].w == SANE_TRUE) ? FALSE : TRUE; /* skip warmup process? */ RTS_Debug->warmup = (s->aValues[opt_nowarmup].w == SANE_TRUE) ? FALSE : TRUE; /* save debugging images? */ RTS_Debug->SaveCalibFile = (s->aValues[opt_dbgimages].w == SANE_TRUE) ? TRUE : FALSE; /* Get image coordinates in milimeters */ coords.left = s->aValues[opt_tlx].w; coords.top = s->aValues[opt_tly].w; coords.width = s->aValues[opt_brx].w; coords.height = s->aValues[opt_bry].w; /* Validate coords */ if (Translate_coords (&coords) == SANE_STATUS_GOOD) { /* Stop previusly started scan */ RTS_Scanner_StopScan (device, TRUE); s->ScanParams.scantype = source; s->ScanParams.colormode = colormode; s->ScanParams.resolution_x = res; s->ScanParams.resolution_y = res; s->ScanParams.channel = channel; memcpy (&s->ScanParams.coords, &coords, sizeof (struct st_coords)); Set_Coordinates (source, res, &s->ScanParams.coords); /* emulating depth? */ if ((s->cnv.real_depth == FALSE) && (depth < 16) && (RTS_Debug->EnableGamma == TRUE)) { /* In order to improve image quality, we will scan at 16bits if we are using gamma correction */ s->cnv.depth = depth; s->ScanParams.depth = 16; } else { s->ScanParams.depth = depth; s->cnv.depth = -1; } /* set scanning parameters */ if (RTS_Scanner_SetParams (device, &s->ScanParams) == OK) { /* Start scanning process */ if (RTS_Scanner_StartScan (device) == OK) { /* Allocate buffer to read one line */ s->mylin = 0; rst = img_buffers_alloc (s, bytesperline); } } } } else rst = SANE_STATUS_COVER_OPEN; } DBG (DBG_FNC, "- sane_start: %i\n", rst); return rst; } SANE_Status sane_read (SANE_Handle h, SANE_Byte * buf, SANE_Int maxlen, SANE_Int * len) { SANE_Status rst = SANE_STATUS_GOOD; TScanner *s = (TScanner *) h; DBG (DBG_FNC, "+ sane_read\n"); if ((s != NULL) && (buf != NULL) && (len != NULL)) { /* nothing has been read at the moment */ *len = 0; /* if we read all the lines return EOF */ if ((s->mylin == s->ScanParams.coords.height) || (device->status->cancel == TRUE)) { rst = (device->status->cancel == TRUE) ? SANE_STATUS_CANCELLED : SANE_STATUS_EOF; RTS_Scanner_StopScan (device, FALSE); img_buffers_free (s); } else { SANE_Int emul_len, emul_maxlen; SANE_Int thwidth, transferred, bufflength; SANE_Byte *buffer, *pbuffer; emul_len = 0; if (s->cnv.depth != -1) emul_maxlen = maxlen * (s->ScanParams.depth / s->cnv.depth); else emul_maxlen = maxlen; /* if grayscale emulation is enabled check that retrieved data is multiple of three */ if (s->cnv.colormode == CM_GRAY) { SANE_Int chn_size, rest; chn_size = (s->ScanParams.depth > 8) ? 2 : 1; rest = emul_maxlen % (3 * chn_size); if (rest != 0) emul_maxlen -= rest; } /* this is important to keep lines alignment in lineart mode */ if (s->cnv.colormode == CM_LINEART) emul_maxlen = s->ScanParams.coords.width; /* if we are emulating depth, we scan at 16bit when frontend waits for 8bit data. Next buffer will be used to retrieve data from scanner prior to convert to 8 bits depth */ buffer = (SANE_Byte *) malloc (emul_maxlen * sizeof (SANE_Byte)); if (buffer != NULL) { pbuffer = buffer; /* get bytes per line */ if (s->ScanParams.colormode != CM_LINEART) { thwidth = s->ScanParams.coords.width * ((s->ScanParams.depth > 8) ? 2 : 1); if (s->ScanParams.colormode == CM_COLOR) thwidth *= 3; /* three channels */ } else thwidth = (s->ScanParams.coords.width + 7) / 8; /* read as many lines the buffer may contain and while there are lines to be read */ while ((emul_len < emul_maxlen) && (s->mylin < s->ScanParams.coords.height)) { /* Is there any data waiting for being passed ? */ if (s->rest_amount != 0) { /* copy to buffer as many bytes as we can */ bufflength = min (emul_maxlen - emul_len, s->rest_amount); memcpy (pbuffer, s->rest, bufflength); emul_len += bufflength; pbuffer += bufflength; s->rest_amount -= bufflength; if (s->rest_amount == 0) s->mylin++; } else { /* read from scanner up to one line */ if (Read_Image (device, bytesperline, s->image, &transferred) != OK) { /* error, exit function */ rst = SANE_STATUS_EOF; break; } /* is there any data? */ if (transferred != 0) { /* copy to buffer as many bytes as we can */ bufflength = min (emul_maxlen - emul_len, thwidth); memcpy (pbuffer, s->image, bufflength); emul_len += bufflength; pbuffer += bufflength; /* the rest will be copied to s->rest buffer */ if (bufflength < thwidth) { s->rest_amount = thwidth - bufflength; memcpy (s->rest, s->image + bufflength, s->rest_amount); } else s->mylin++; } else break; } } /* while */ /* process buffer before sending to frontend */ if ((emul_len > 0) && (rst != SANE_STATUS_EOF)) { /* at this point ... buffer : contains retrieved image emul_len: contains size in bytes of retrieved image after this code ... buf : will contain postprocessed image len : will contain size in bytes of postprocessed image */ /* apply gamma if neccesary */ if (RTS_Debug->EnableGamma == TRUE) gamma_apply (s, buffer, emul_len, s->ScanParams.depth); /* if we are scanning negatives, let's invert colors */ if (s->ScanParams.scantype == ST_NEG) { if (s->cnv.negative == FALSE) Color_Negative (buffer, emul_len, s->ScanParams.depth); } else if (s->cnv.negative != FALSE) Color_Negative (buffer, emul_len, s->ScanParams.depth); /* emulating grayscale ? */ if (s->cnv.colormode == CM_GRAY) { Color_to_Gray (buffer, emul_len, s->ScanParams.depth); emul_len /= 3; } /* emulating depth */ if (s->cnv.depth != -1) { switch (s->cnv.depth) { /* case 1: treated separately as lineart */ /*case 12: in the future */ case 8: Depth_16_to_8 (buffer, emul_len, buffer); emul_len /= 2; break; } } /* lineart mode ? */ if (s->cnv.colormode == CM_LINEART) { /* I didn't see any scanner supporting lineart mode. Windows drivers scan in grayscale and then convert image to lineart so let's perform convertion */ SANE_Int rest = emul_len % 8; Gray_to_Lineart (buffer, emul_len, s->cnv.threshold); emul_len /= 8; if (rest > 0) emul_len++; } /* copy postprocessed image */ *len = emul_len; memcpy (buf, buffer, *len); } free (buffer); } } } else rst = SANE_STATUS_EOF; DBG (DBG_FNC, "- sane_read: %s\n", sane_strstatus (rst)); return rst; } void sane_cancel (SANE_Handle h) { DBG (DBG_FNC, "> sane_cancel\n"); /* silence gcc */ h = h; device->status->cancel = TRUE; } SANE_Status sane_set_io_mode (SANE_Handle handle, SANE_Bool non_blocking) { DBG (DBG_FNC, "> sane_set_io_mode\n"); /* silence gcc */ handle = handle; non_blocking = non_blocking; return SANE_STATUS_UNSUPPORTED; } SANE_Status sane_get_select_fd (SANE_Handle handle, SANE_Int * fd) { DBG (DBG_FNC, "> sane_get_select_fd\n"); /* silence gcc */ handle = handle; fd = fd; return SANE_STATUS_UNSUPPORTED; } void sane_close (SANE_Handle h) { TScanner *scanner = (TScanner *) h; DBG (DBG_FNC, "- sane_close...\n"); /* stop previus scans */ RTS_Scanner_StopScan (device, TRUE); /* close usb */ sanei_usb_close (device->usb_handle); /* free scanner internal variables */ RTS_Scanner_End (device); /* free RTS enviroment */ RTS_Free (device); /* free backend variables */ if (scanner != NULL) { options_free (scanner); img_buffers_free (scanner); } } void sane_exit (void) { /* free device list memory */ if (_pSaneDevList) { TDevListEntry *pDev, *pNext; for (pDev = _pFirstSaneDev; pDev; pDev = pNext) { pNext = pDev->pNext; /* pDev->dev.name is the same pointer that pDev->devname */ free (pDev->devname); free (pDev); } _pFirstSaneDev = NULL; free (_pSaneDevList); _pSaneDevList = NULL; } }