kopia lustrzana https://gitlab.com/sane-project/backends
				
				
				
			
		
			
				
	
	
		
			454 wiersze
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			454 wiersze
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
/* sane - Scanner Access Now Easy.
 | 
						|
   Copyright (C) 1997 Jeffrey S. Freedman
 | 
						|
   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.
 | 
						|
 | 
						|
   This file provides generic configuration support.  */
 | 
						|
 | 
						|
#include "../include/sane/config.h"
 | 
						|
 | 
						|
#include <ctype.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
#include <sys/param.h>
 | 
						|
 | 
						|
#include "../include/sane/sanei.h"
 | 
						|
#include "../include/sane/sanei_config.h"
 | 
						|
 | 
						|
#define BACKEND_NAME	sanei_config
 | 
						|
#include "../include/sane/sanei_debug.h"
 | 
						|
 | 
						|
#ifndef PATH_MAX
 | 
						|
# define PATH_MAX	1024
 | 
						|
#endif
 | 
						|
 | 
						|
#if defined(_WIN32) || defined(HAVE_OS2_H)
 | 
						|
# define DIR_SEP	";"
 | 
						|
# define PATH_SEP	'\\'
 | 
						|
#else
 | 
						|
# define DIR_SEP	":"
 | 
						|
# define PATH_SEP	'/'
 | 
						|
#endif
 | 
						|
 | 
						|
#define DEFAULT_DIRS	"." DIR_SEP STRINGIFY(PATH_SANE_CONFIG_DIR)
 | 
						|
 | 
						|
#ifdef __BEOS__
 | 
						|
#include <FindDirectory.h>
 | 
						|
#endif
 | 
						|
 | 
						|
static char *dir_list;
 | 
						|
 | 
						|
const char *
 | 
						|
sanei_config_get_paths ()
 | 
						|
{
 | 
						|
#ifdef __BEOS__
 | 
						|
  char result[PATH_MAX];
 | 
						|
#endif
 | 
						|
  void *mem;
 | 
						|
  char *dlist;
 | 
						|
  size_t len;
 | 
						|
 | 
						|
  if (!dir_list)
 | 
						|
    {
 | 
						|
      DBG_INIT();
 | 
						|
 | 
						|
      dlist = getenv ("SANE_CONFIG_DIR");
 | 
						|
      if (dlist)
 | 
						|
	dir_list = strdup (dlist);
 | 
						|
#ifdef __BEOS__
 | 
						|
      /* ~/config/settings/SANE takes precedence over /etc/sane.d/ */
 | 
						|
      if (!dir_list)
 | 
						|
	{
 | 
						|
	  if (find_directory(B_USER_SETTINGS_DIRECTORY, 0, true, result, PATH_MAX) == B_OK)
 | 
						|
	    {
 | 
						|
	      strcat(result,"/SANE");
 | 
						|
	      strcat(result,DIR_SEP); /* do append the default ones */
 | 
						|
	      dir_list = strdup (result);
 | 
						|
	    }
 | 
						|
	}
 | 
						|
#endif
 | 
						|
      if (dir_list)
 | 
						|
	{
 | 
						|
	  len = strlen (dir_list);
 | 
						|
	  if ((len > 0) && (dir_list[len - 1] == DIR_SEP[0]))
 | 
						|
	    {
 | 
						|
	      /* append default search directories: */
 | 
						|
	      mem = malloc (len + sizeof (DEFAULT_DIRS));
 | 
						|
	      memcpy (mem, dir_list, len);
 | 
						|
	      memcpy ((char *) mem + len, DEFAULT_DIRS, sizeof (DEFAULT_DIRS));
 | 
						|
	      free (dir_list);
 | 
						|
	      dir_list = mem;
 | 
						|
	    }
 | 
						|
	}
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  /* Create a copy, since we might call free on it */
 | 
						|
	  dir_list = strdup (DEFAULT_DIRS);
 | 
						|
	}
 | 
						|
    }
 | 
						|
  DBG (5, "sanei_config_get_paths: using config directories  %s\n", dir_list);
 | 
						|
 | 
						|
  return dir_list;
 | 
						|
}
 | 
						|
 | 
						|
FILE *
 | 
						|
sanei_config_open (const char *filename)
 | 
						|
{
 | 
						|
  char *next, *dir, result[PATH_MAX];
 | 
						|
  const char *cfg_dir_list;
 | 
						|
  FILE *fp;
 | 
						|
  char *copy;
 | 
						|
 | 
						|
  cfg_dir_list = sanei_config_get_paths ();
 | 
						|
  if (!cfg_dir_list)
 | 
						|
    {
 | 
						|
      DBG(2, "sanei_config_open: could not find config file `%s'\n", filename);
 | 
						|
      return NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  copy = strdup (cfg_dir_list);
 | 
						|
 | 
						|
  for (next = copy; (dir = strsep (&next, DIR_SEP)) != 0; )
 | 
						|
    {
 | 
						|
      snprintf (result, sizeof (result), "%s%c%s", dir, PATH_SEP, filename);
 | 
						|
      DBG(4, "sanei_config_open: attempting to open `%s'\n", result);
 | 
						|
      fp = fopen (result, "r");
 | 
						|
      if (fp)
 | 
						|
	{
 | 
						|
	  DBG(3, "sanei_config_open: using file `%s'\n", result);
 | 
						|
	  break;
 | 
						|
	}
 | 
						|
    }
 | 
						|
  free (copy);
 | 
						|
 | 
						|
  if (!fp)
 | 
						|
    DBG(2, "sanei_config_open: could not find config file `%s'\n", filename);
 | 
						|
 | 
						|
  return fp;
 | 
						|
}
 | 
						|
 | 
						|
const char *
 | 
						|
sanei_config_skip_whitespace (const char *str)
 | 
						|
{
 | 
						|
  while (str && *str && isspace (*str))
 | 
						|
    ++str;
 | 
						|
  return str;
 | 
						|
}
 | 
						|
 | 
						|
const char *
 | 
						|
sanei_config_get_string (const char *str, char **string_const)
 | 
						|
{
 | 
						|
  const char *start;
 | 
						|
  size_t len;
 | 
						|
 | 
						|
  str = sanei_config_skip_whitespace (str);
 | 
						|
 | 
						|
  if (*str == '"')
 | 
						|
    {
 | 
						|
      start = ++str;
 | 
						|
      while (*str && *str != '"')
 | 
						|
	++str;
 | 
						|
      len = str - start;
 | 
						|
      if (*str == '"')
 | 
						|
	++str;
 | 
						|
      else
 | 
						|
	start = 0;		/* final double quote is missing */
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      start = str;
 | 
						|
      while (*str && !isspace (*str))
 | 
						|
	++str;
 | 
						|
      len = str - start;
 | 
						|
    }
 | 
						|
  if (start)
 | 
						|
    *string_const = strndup (start, len);
 | 
						|
  else
 | 
						|
    *string_const = 0;
 | 
						|
  return str;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
sanei_config_read (char *str, int n, FILE *stream)
 | 
						|
{
 | 
						|
   char* rc;
 | 
						|
   char* start;
 | 
						|
   int   len;
 | 
						|
 | 
						|
      /* read line from stream */
 | 
						|
   rc = fgets( str, n, stream);
 | 
						|
   if (rc == NULL)
 | 
						|
      return NULL;
 | 
						|
 | 
						|
      /* remove ending whitespaces */
 | 
						|
   len = strlen( str);
 | 
						|
   while( (0 < len) && (isspace( str[--len])) )
 | 
						|
      str[len] = '\0';
 | 
						|
 | 
						|
      /* remove starting whitespaces */
 | 
						|
   start = str;
 | 
						|
   while( isspace( *start))
 | 
						|
      start++;
 | 
						|
 | 
						|
   if (start != str) 
 | 
						|
      do {
 | 
						|
         *str++ = *start++;
 | 
						|
      } while( *str);
 | 
						|
 | 
						|
   return rc;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
SANE_Status
 | 
						|
sanei_configure_attach (const char *config_file, SANEI_Config * config,
 | 
						|
			SANE_Status (*attach) (SANEI_Config * config,
 | 
						|
					       const char *devname))
 | 
						|
{
 | 
						|
  SANE_Char line[PATH_MAX];
 | 
						|
  SANE_Char *token, *string;
 | 
						|
  SANE_Int len;
 | 
						|
  const char *lp, *lp2;
 | 
						|
  FILE *fp;
 | 
						|
  SANE_Status status = SANE_STATUS_GOOD;
 | 
						|
  int i, j, count;
 | 
						|
  void *value = NULL;
 | 
						|
  int size=0;
 | 
						|
  SANE_Bool found;
 | 
						|
  SANE_Word *wa;
 | 
						|
  SANE_Bool *ba;
 | 
						|
 | 
						|
  DBG (3, "sanei_configure_attach: start\n");
 | 
						|
 | 
						|
  /* open configuration file */
 | 
						|
  fp = sanei_config_open (config_file);
 | 
						|
  if (!fp)
 | 
						|
    {
 | 
						|
      DBG (2, "sanei_configure_attach: couldn't access %s\n", config_file);
 | 
						|
      DBG (3, "sanei_configure_attach: exit\n");
 | 
						|
      return SANE_STATUS_ACCESS_DENIED;
 | 
						|
    }
 | 
						|
 | 
						|
  /* loop reading the configuration file, all line beginning by "option " are
 | 
						|
   * parsed for value to store in configuration structure, other line are 
 | 
						|
   * used are device to try to attach
 | 
						|
   */
 | 
						|
  while (sanei_config_read (line, PATH_MAX, fp) && status == SANE_STATUS_GOOD)
 | 
						|
    {
 | 
						|
      /* skip white spaces at beginning of line */
 | 
						|
      lp = sanei_config_skip_whitespace (line);
 | 
						|
 | 
						|
      /* skip empty lines */
 | 
						|
      if (*lp == 0)
 | 
						|
	continue;
 | 
						|
 | 
						|
      /* skip comment line */
 | 
						|
      if (line[0] == '#')
 | 
						|
	continue;
 | 
						|
 | 
						|
      len = strlen (line);
 | 
						|
 | 
						|
      /* delete newline characters at end */
 | 
						|
      if (line[len - 1] == '\n')
 | 
						|
	line[--len] = '\0';
 | 
						|
 | 
						|
      lp2 = lp;
 | 
						|
 | 
						|
      /* to ensure maximum compatibility, we accept line like:
 | 
						|
       * option "option_name" "option_value"
 | 
						|
       * "option_name" "option_value" 
 | 
						|
       * So we parse the line 2 time to find an option */
 | 
						|
      /* check if it is an option */
 | 
						|
      lp = sanei_config_get_string (lp, &token);
 | 
						|
      if (strncmp (token, "option", 6) == 0)
 | 
						|
	{
 | 
						|
	  /* skip the "option" token */
 | 
						|
	  free (token);
 | 
						|
	  lp = sanei_config_get_string (lp, &token);
 | 
						|
	}
 | 
						|
 | 
						|
      /* search for a matching descriptor */
 | 
						|
      i = 0;
 | 
						|
      found = SANE_FALSE;
 | 
						|
      while (config!=NULL && i < config->count && !found)
 | 
						|
	{
 | 
						|
	  if (strcmp (config->descriptors[i]->name, token) == 0)
 | 
						|
	    {
 | 
						|
	      found = SANE_TRUE;
 | 
						|
	      switch (config->descriptors[i]->type)
 | 
						|
		{
 | 
						|
		case SANE_TYPE_INT:
 | 
						|
		  size=config->descriptors[i]->size;
 | 
						|
		  value = malloc (size);
 | 
						|
		  wa = (SANE_Word *) value;
 | 
						|
		  count = config->descriptors[i]->size / sizeof (SANE_Word);
 | 
						|
		  for (j = 0; j < count; j++)
 | 
						|
		    {
 | 
						|
		      lp = sanei_config_get_string (lp, &string);
 | 
						|
		      if (string == NULL)
 | 
						|
			{
 | 
						|
			  DBG (2,
 | 
						|
			       "sanei_configure_attach: couldn't find a string to parse");
 | 
						|
			  return SANE_STATUS_INVAL;
 | 
						|
			}
 | 
						|
		      wa[j] = strtol (string, NULL, 0);
 | 
						|
		      free (string);
 | 
						|
		    }
 | 
						|
		  break;
 | 
						|
		case SANE_TYPE_BOOL:
 | 
						|
		  size=config->descriptors[i]->size;
 | 
						|
		  value = malloc (size);
 | 
						|
		  ba = (SANE_Bool *) value;
 | 
						|
		  count = config->descriptors[i]->size / sizeof (SANE_Bool);
 | 
						|
		  for (j = 0; j < count; j++)
 | 
						|
		    {
 | 
						|
		      lp = sanei_config_get_string (lp, &string);
 | 
						|
		      if (string == NULL)
 | 
						|
			{
 | 
						|
			  DBG (2,
 | 
						|
			       "sanei_configure_attach: couldn't find a string to parse");
 | 
						|
			  return SANE_STATUS_INVAL;
 | 
						|
			}
 | 
						|
		      if ((strcmp (string, "1") == 0)
 | 
						|
			  || (strcmp (string, "true") == 0))
 | 
						|
			{
 | 
						|
			  ba[j] = SANE_TRUE;
 | 
						|
			}
 | 
						|
		      else
 | 
						|
			{
 | 
						|
			  if ((strcmp (string, "0") == 0)
 | 
						|
			      || (strcmp (string, "false") == 0))
 | 
						|
			    ba[j] = SANE_FALSE;
 | 
						|
			  else
 | 
						|
			    {
 | 
						|
			      DBG (2,
 | 
						|
				   "sanei_configure_attach: couldn't find a valid boolean value");
 | 
						|
			      return SANE_STATUS_INVAL;
 | 
						|
			    }
 | 
						|
			}
 | 
						|
		      free (string);
 | 
						|
		    }
 | 
						|
		  break;
 | 
						|
		case SANE_TYPE_FIXED:
 | 
						|
		  size=config->descriptors[i]->size;
 | 
						|
		  value = malloc (size);
 | 
						|
		  wa = (SANE_Word *) value;
 | 
						|
		  count = config->descriptors[i]->size / sizeof (SANE_Word);
 | 
						|
		  for (j = 0; j < count; j++)
 | 
						|
		    {
 | 
						|
		      lp = sanei_config_get_string (lp, &string);
 | 
						|
		      if (string == NULL)
 | 
						|
			{
 | 
						|
			  DBG (2,
 | 
						|
			       "sanei_configure_attach: couldn't find a string to parse");
 | 
						|
			  return SANE_STATUS_INVAL;
 | 
						|
			}
 | 
						|
		      wa[j] = SANE_FIX(strtod (string, NULL));
 | 
						|
		      free (string);
 | 
						|
		    }
 | 
						|
		  break;
 | 
						|
		case SANE_TYPE_STRING:
 | 
						|
		  sanei_config_get_string (lp, &string);
 | 
						|
		  if (string == NULL)
 | 
						|
		    {
 | 
						|
		      DBG (2,
 | 
						|
			   "sanei_configure_attach: couldn't find a string value to parse");
 | 
						|
		      return SANE_STATUS_INVAL;
 | 
						|
		    }
 | 
						|
		  value = string;
 | 
						|
		  size=strlen(string)+1;
 | 
						|
		  if(size>config->descriptors[i]->size)
 | 
						|
		  {
 | 
						|
			  size=config->descriptors[i]->size-1;
 | 
						|
			  string[size]=0;
 | 
						|
		  }
 | 
						|
		  break;
 | 
						|
		default:
 | 
						|
		  DBG (1,
 | 
						|
		       "sanei_configure_attach: incorrect type %d for option %s, skipping option ...\n",
 | 
						|
		       config->descriptors[i]->type,
 | 
						|
		       config->descriptors[i]->name);
 | 
						|
		}
 | 
						|
	      
 | 
						|
	      /* check decoded value */
 | 
						|
	      status = sanei_check_value (config->descriptors[i], value);
 | 
						|
 | 
						|
	      /* if value OK, copy it in configuration struct */
 | 
						|
	      if (status == SANE_STATUS_GOOD)
 | 
						|
		{
 | 
						|
		  memcpy (config->values[i], value, size);
 | 
						|
		}
 | 
						|
	      if (value != NULL)
 | 
						|
		{
 | 
						|
		  free (value);
 | 
						|
		  value = NULL;
 | 
						|
		}
 | 
						|
	    }
 | 
						|
	  if (status != SANE_STATUS_GOOD)
 | 
						|
	    {
 | 
						|
	      DBG (1,
 | 
						|
		   "sanei_configure_attach: failed to parse option '%s', line '%s'\n",
 | 
						|
		   token, line);
 | 
						|
	    }
 | 
						|
	  i++;
 | 
						|
	}
 | 
						|
      free (token);
 | 
						|
 | 
						|
      /* not detected as an option, so we call the attach function
 | 
						|
       * with it */
 | 
						|
      if (!found && status == SANE_STATUS_GOOD)
 | 
						|
	{
 | 
						|
	  /* if not an option, try to attach */
 | 
						|
	  /* to avoid every backend to depend on scsi and usb functions
 | 
						|
	   * we call back the backend for attach. In turn it will call
 | 
						|
	   * sanei_usb_attach_matching_devices, sanei_config_attach_matching_devices
 | 
						|
	   * or other. This means 2 callback functions per backend using this 
 | 
						|
	   * function. */
 | 
						|
	  DBG (3, "sanei_configure_attach: trying to attach with '%s'\n",
 | 
						|
	       lp2);
 | 
						|
	  if(attach!=NULL)
 | 
						|
	  	attach (config, lp2);
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  fclose (fp);
 | 
						|
  DBG (3, "sanei_configure_attach: exit\n");
 | 
						|
  return status;
 | 
						|
}
 |