kopia lustrzana https://gitlab.com/sane-project/backends
				
				
				
			
		
			
				
	
	
		
			346 wiersze
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			346 wiersze
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
/* sane - Scanner Access Now Easy.
 | 
						|
   Copyright (C) 1997 David Mosberger-Tang
 | 
						|
   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.  */
 | 
						|
 | 
						|
#include "../include/sane/config.h"
 | 
						|
 | 
						|
#include <ctype.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
#include "../include/sane/sane.h"
 | 
						|
#include "../include/sane/sanei_wire.h"
 | 
						|
#include "../include/sane/sanei_codec_ascii.h"
 | 
						|
 | 
						|
static const char *hexdigit = "0123456789abcdef";
 | 
						|
 | 
						|
static void
 | 
						|
skip_ws (Wire *w)
 | 
						|
{
 | 
						|
  while (1)
 | 
						|
    {
 | 
						|
      sanei_w_space (w, 1);
 | 
						|
      if (w->status != 0)
 | 
						|
	return;
 | 
						|
 | 
						|
      if (!isspace (*w->buffer.curr))
 | 
						|
	return;
 | 
						|
 | 
						|
      ++w->buffer.curr;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static unsigned
 | 
						|
get_digit (Wire *w)
 | 
						|
{
 | 
						|
  unsigned digit;
 | 
						|
 | 
						|
  sanei_w_space (w, 1);
 | 
						|
  digit = tolower(*w->buffer.curr++) - '0';
 | 
						|
  if (digit > 9)
 | 
						|
    digit -= 'a' - ('9' + 1);
 | 
						|
  if (digit > 0xf)
 | 
						|
    {
 | 
						|
      w->status = EINVAL;
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
  return digit;
 | 
						|
}
 | 
						|
 | 
						|
static SANE_Byte
 | 
						|
get_byte (Wire *w)
 | 
						|
{
 | 
						|
  return get_digit (w) << 4 | get_digit (w);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
ascii_w_byte (Wire *w, void *v)
 | 
						|
{
 | 
						|
  SANE_Byte *b = v;
 | 
						|
 | 
						|
  switch (w->direction)
 | 
						|
    {
 | 
						|
    case WIRE_ENCODE:
 | 
						|
      sanei_w_space (w, 3);
 | 
						|
      *w->buffer.curr++ = hexdigit[(*b >> 4) & 0x0f];
 | 
						|
      *w->buffer.curr++ = hexdigit[(*b >> 0) & 0x0f];
 | 
						|
      *w->buffer.curr++ = '\n';
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_DECODE:
 | 
						|
      skip_ws (w);
 | 
						|
      *b = get_byte (w);
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_FREE:
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
ascii_w_char (Wire *w, void *v)
 | 
						|
{
 | 
						|
  SANE_Char *c = v;
 | 
						|
 | 
						|
  switch (w->direction)
 | 
						|
    {
 | 
						|
    case WIRE_ENCODE:
 | 
						|
      sanei_w_space (w, 5);
 | 
						|
      *w->buffer.curr++ = '\'';
 | 
						|
      if (*c == '\'' || *c == '\\')
 | 
						|
	*w->buffer.curr++ = '\\';
 | 
						|
      *w->buffer.curr++ = *c;
 | 
						|
      *w->buffer.curr++ = '\'';
 | 
						|
      *w->buffer.curr++ = '\n';
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_DECODE:
 | 
						|
      sanei_w_space (w, 4);
 | 
						|
      if (*w->buffer.curr++ != '\'')
 | 
						|
	{
 | 
						|
	  w->status = EINVAL;
 | 
						|
	  return;
 | 
						|
	}
 | 
						|
      *c = *w->buffer.curr++;
 | 
						|
      if (*c == '\\')
 | 
						|
	{
 | 
						|
	  sanei_w_space (w, 2);
 | 
						|
	  *c = *w->buffer.curr++;
 | 
						|
	}
 | 
						|
      if (*w->buffer.curr++ != '\'')
 | 
						|
	{
 | 
						|
	  w->status = EINVAL;
 | 
						|
	  return;
 | 
						|
	}
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_FREE:
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
ascii_w_string (Wire *w, void *v)
 | 
						|
{
 | 
						|
  size_t len, alloced_len;
 | 
						|
  SANE_String *s = v;
 | 
						|
  char * str, ch;
 | 
						|
  int done;
 | 
						|
 | 
						|
  switch (w->direction)
 | 
						|
    {
 | 
						|
    case WIRE_ENCODE:
 | 
						|
      if (*s)
 | 
						|
	{
 | 
						|
	  sanei_w_space (w, 1);
 | 
						|
	  *w->buffer.curr++ = '"';
 | 
						|
	  str = *s;
 | 
						|
	  while ((ch = *str++))
 | 
						|
	    {
 | 
						|
	      sanei_w_space (w, 2);
 | 
						|
	      if (ch == '"' || ch == '\\')
 | 
						|
		*w->buffer.curr++ = '\\';
 | 
						|
	      *w->buffer.curr++ = ch;
 | 
						|
	    }
 | 
						|
	  *w->buffer.curr++ = '"';
 | 
						|
	}
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  sanei_w_space (w, 5);
 | 
						|
	  *w->buffer.curr++ = '(';
 | 
						|
	  *w->buffer.curr++ = 'n';
 | 
						|
	  *w->buffer.curr++ = 'i';
 | 
						|
	  *w->buffer.curr++ = 'l';
 | 
						|
	  *w->buffer.curr++ = ')';
 | 
						|
	}
 | 
						|
      sanei_w_space (w, 1);
 | 
						|
      *w->buffer.curr++ = '\n';
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_DECODE:
 | 
						|
      skip_ws (w);
 | 
						|
      sanei_w_space (w, 1);
 | 
						|
      ch = *w->buffer.curr++;
 | 
						|
      if (ch == '"')
 | 
						|
	{
 | 
						|
	  alloced_len = len = 0;
 | 
						|
	  str = 0;
 | 
						|
	  done = 0;
 | 
						|
	  do
 | 
						|
	    {
 | 
						|
	      sanei_w_space (w, 1);
 | 
						|
	      if (w->status != 0)
 | 
						|
		return;
 | 
						|
 | 
						|
	      ch = *w->buffer.curr++;
 | 
						|
	      if (ch == '"')
 | 
						|
		done = 1;
 | 
						|
 | 
						|
	      if (ch == '\\')
 | 
						|
		{
 | 
						|
		  sanei_w_space (w, 1);
 | 
						|
		  ch = *w->buffer.curr++;
 | 
						|
		}
 | 
						|
 | 
						|
	      if (len >= alloced_len)
 | 
						|
		{
 | 
						|
		  alloced_len += 1024;
 | 
						|
		  if (!str)
 | 
						|
		    str = malloc (alloced_len);
 | 
						|
		  else
 | 
						|
		    str = realloc (str, alloced_len);
 | 
						|
 | 
						|
		  if (str == 0)
 | 
						|
		    {
 | 
						|
		      /* Malloc failed, so return an error. */
 | 
						|
		      w->status = ENOMEM;
 | 
						|
		      return;
 | 
						|
		    }
 | 
						|
		}
 | 
						|
	      str[len++] = ch;
 | 
						|
	    }
 | 
						|
	  while (!done);
 | 
						|
 | 
						|
	  str[len - 1] = '\0';
 | 
						|
	  *s = realloc (str, len);
 | 
						|
 | 
						|
	  if (*s == 0)
 | 
						|
	    {
 | 
						|
	      /* Malloc failed, so return an error. */
 | 
						|
	      w->status = ENOMEM;
 | 
						|
	      return;
 | 
						|
	    }
 | 
						|
	}
 | 
						|
      else if (ch == '(')
 | 
						|
	{
 | 
						|
	  sanei_w_space (w, 4);
 | 
						|
	  if (   *w->buffer.curr++ != 'n'
 | 
						|
	      || *w->buffer.curr++ != 'i'
 | 
						|
	      || *w->buffer.curr++ != 'l'
 | 
						|
	      || *w->buffer.curr++ != ')')
 | 
						|
	    {
 | 
						|
	      w->status = EINVAL;
 | 
						|
	      return;
 | 
						|
	    }
 | 
						|
	  *s = 0;
 | 
						|
	}
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  w->status = EINVAL;
 | 
						|
	  return;
 | 
						|
	}
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_FREE:
 | 
						|
      if (*s)
 | 
						|
	free (*s);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
ascii_w_word (Wire *w, void *v)
 | 
						|
{
 | 
						|
  SANE_Word val, *word = v;
 | 
						|
  int i, is_negative = 0;
 | 
						|
  char buf[16];
 | 
						|
 | 
						|
  switch (w->direction)
 | 
						|
    {
 | 
						|
    case WIRE_ENCODE:
 | 
						|
      val = *word;
 | 
						|
      i = sizeof (buf) - 1;
 | 
						|
      if (val < 0)
 | 
						|
	{
 | 
						|
	  is_negative = 1;
 | 
						|
	  val = -val;
 | 
						|
	}
 | 
						|
      do
 | 
						|
	{
 | 
						|
	  buf[i--] = '0' + (val % 10);
 | 
						|
	  val /= 10;
 | 
						|
	}
 | 
						|
      while (val);
 | 
						|
      if (is_negative)
 | 
						|
	buf[i--] = '-';
 | 
						|
 | 
						|
      sanei_w_space (w, sizeof (buf) - i);
 | 
						|
      memcpy (w->buffer.curr, buf + i + 1, sizeof (buf) - i - 1);
 | 
						|
      w->buffer.curr += sizeof (buf) - i - 1;
 | 
						|
      *w->buffer.curr++ = '\n';
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_DECODE:
 | 
						|
      skip_ws (w);
 | 
						|
      val = 0;
 | 
						|
      sanei_w_space (w, 1);
 | 
						|
      if (*w->buffer.curr == '-')
 | 
						|
	{
 | 
						|
	  is_negative = 1;
 | 
						|
	  ++w->buffer.curr;
 | 
						|
	}
 | 
						|
      while (1)
 | 
						|
	{
 | 
						|
	  sanei_w_space (w, 1);
 | 
						|
	  if (w->status != 0)
 | 
						|
	    return;
 | 
						|
 | 
						|
	  if (!isdigit (*w->buffer.curr))
 | 
						|
	    break;
 | 
						|
 | 
						|
	  val = 10*val + (*w->buffer.curr++ - '0');
 | 
						|
	}
 | 
						|
      *word = is_negative ? -val : val;
 | 
						|
      break;
 | 
						|
 | 
						|
    case WIRE_FREE:
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
sanei_codec_ascii_init (Wire *w)
 | 
						|
{
 | 
						|
  w->codec.w_byte = ascii_w_byte;
 | 
						|
  w->codec.w_char = ascii_w_char;
 | 
						|
  w->codec.w_word = ascii_w_word;
 | 
						|
  w->codec.w_string = ascii_w_string;
 | 
						|
}
 |