1999-08-09 18:06:01 +00:00
|
|
|
/* sane - Scanner Access Now Easy.
|
|
|
|
Copyright (C) 1996, 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.
|
|
|
|
|
|
|
|
This file implements a dynamic linking based SANE meta backend. It
|
|
|
|
allows managing an arbitrary number of SANE backends by using
|
|
|
|
dynamic linking to load backends on demand. */
|
|
|
|
|
2001-04-16 12:21:41 +00:00
|
|
|
/* Please increase version number with every change
|
|
|
|
(don't forget to update dll.desc) */
|
2001-10-02 20:35:08 +00:00
|
|
|
#define DLL_VERSION "1.0.4"
|
2001-04-13 17:04:51 +00:00
|
|
|
|
1999-08-09 18:06:01 +00:00
|
|
|
#ifdef _AIX
|
2000-08-12 15:11:46 +00:00
|
|
|
# include "lalloca.h" /* MUST come first for AIX! */
|
1999-08-09 18:06:01 +00:00
|
|
|
#endif
|
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
#include "sane/config.h"
|
|
|
|
#include "lalloca.h"
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#if defined(HAVE_DLOPEN) && defined(HAVE_DLFCN_H)
|
|
|
|
# include <dlfcn.h>
|
|
|
|
|
|
|
|
/* Older versions of dlopen() don't define RTLD_NOW and RTLD_LAZY.
|
|
|
|
They all seem to use a mode of 1 to indicate RTLD_NOW and some do
|
|
|
|
not support RTLD_LAZY at all. Hence, unless defined, we define
|
|
|
|
both macros as 1 to play it safe. */
|
|
|
|
# ifndef RTLD_NOW
|
2000-08-12 15:11:46 +00:00
|
|
|
# define RTLD_NOW 1
|
1999-08-09 18:06:01 +00:00
|
|
|
# endif
|
|
|
|
# ifndef RTLD_LAZY
|
2000-08-12 15:11:46 +00:00
|
|
|
# define RTLD_LAZY 1
|
1999-08-09 18:06:01 +00:00
|
|
|
# endif
|
|
|
|
# define HAVE_DLL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* HP/UX DLL support */
|
|
|
|
#if defined (HAVE_SHL_LOAD) && defined(HAVE_DL_H)
|
|
|
|
# include <dl.h>
|
|
|
|
# define HAVE_DLL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
#include "sane/sane.h"
|
|
|
|
#include "sane/sanei.h"
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
#define BACKEND_NAME dll
|
2000-08-12 15:11:46 +00:00
|
|
|
#include "sane/sanei_backend.h"
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
#ifndef PATH_MAX
|
2000-08-12 15:11:46 +00:00
|
|
|
# define PATH_MAX 1024
|
1999-08-09 18:06:01 +00:00
|
|
|
#endif
|
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
#include "sane/sanei_config.h"
|
1999-08-09 18:06:01 +00:00
|
|
|
#define DLL_CONFIG_FILE "dll.conf"
|
2000-03-05 13:57:25 +00:00
|
|
|
#define DLL_ALIASES_FILE "dll.aliases"
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
enum SANE_Ops
|
|
|
|
{
|
|
|
|
OP_INIT = 0,
|
|
|
|
OP_EXIT,
|
|
|
|
OP_GET_DEVS,
|
|
|
|
OP_OPEN,
|
|
|
|
OP_CLOSE,
|
|
|
|
OP_GET_OPTION_DESC,
|
|
|
|
OP_CTL_OPTION,
|
|
|
|
OP_GET_PARAMS,
|
|
|
|
OP_START,
|
|
|
|
OP_READ,
|
|
|
|
OP_CANCEL,
|
|
|
|
OP_SET_IO_MODE,
|
|
|
|
OP_GET_SELECT_FD,
|
|
|
|
NUM_OPS
|
|
|
|
};
|
|
|
|
|
|
|
|
struct backend
|
|
|
|
{
|
|
|
|
struct backend *next;
|
|
|
|
const char *name;
|
2000-08-12 15:11:46 +00:00
|
|
|
u_int permanent : 1; /* is the backend preloaded? */
|
|
|
|
u_int loaded : 1; /* are the functions available? */
|
|
|
|
u_int inited : 1; /* has the backend been initialized? */
|
|
|
|
void *handle; /* handle returned by dlopen() */
|
1999-08-09 18:06:01 +00:00
|
|
|
void *(*op[NUM_OPS]) ();
|
|
|
|
};
|
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
#define BE_ENTRY(be,func) sane_##be##_##func
|
|
|
|
|
|
|
|
#define PRELOAD_DECL(name) \
|
|
|
|
extern void *BE_ENTRY(name,init) (); \
|
|
|
|
extern void *BE_ENTRY(name,exit) (); \
|
|
|
|
extern void *BE_ENTRY(name,get_devices) (); \
|
|
|
|
extern void *BE_ENTRY(name,open) (); \
|
|
|
|
extern void *BE_ENTRY(name,close) (); \
|
|
|
|
extern void *BE_ENTRY(name,get_option_descriptor) (); \
|
|
|
|
extern void *BE_ENTRY(name,control_option) (); \
|
|
|
|
extern void *BE_ENTRY(name,get_parameters) (); \
|
|
|
|
extern void *BE_ENTRY(name,start) (); \
|
|
|
|
extern void *BE_ENTRY(name,read) (); \
|
|
|
|
extern void *BE_ENTRY(name,cancel) (); \
|
|
|
|
extern void *BE_ENTRY(name,set_io_mode) (); \
|
1999-08-09 18:06:01 +00:00
|
|
|
extern void *BE_ENTRY(name,get_select_fd) ();
|
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
#define PRELOAD_DEFN(name) \
|
|
|
|
{ \
|
|
|
|
0 /* next */, #name, \
|
|
|
|
1 /* permanent */, \
|
|
|
|
1 /* loaded */, \
|
|
|
|
0 /* inited */, \
|
|
|
|
0 /* handle */, \
|
|
|
|
{ \
|
|
|
|
BE_ENTRY(name,init), \
|
|
|
|
BE_ENTRY(name,exit), \
|
|
|
|
BE_ENTRY(name,get_devices), \
|
|
|
|
BE_ENTRY(name,open), \
|
|
|
|
BE_ENTRY(name,close), \
|
|
|
|
BE_ENTRY(name,get_option_descriptor), \
|
|
|
|
BE_ENTRY(name,control_option), \
|
|
|
|
BE_ENTRY(name,get_parameters), \
|
|
|
|
BE_ENTRY(name,start), \
|
|
|
|
BE_ENTRY(name,read), \
|
|
|
|
BE_ENTRY(name,cancel), \
|
|
|
|
BE_ENTRY(name,set_io_mode), \
|
|
|
|
BE_ENTRY(name,get_select_fd) \
|
|
|
|
} \
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#include "dll-preload.c"
|
|
|
|
|
|
|
|
struct meta_scanner
|
|
|
|
{
|
|
|
|
struct backend *be;
|
|
|
|
SANE_Handle handle;
|
|
|
|
};
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
struct alias
|
|
|
|
{
|
|
|
|
struct alias *next;
|
|
|
|
char *oldname;
|
|
|
|
char *newname;
|
|
|
|
};
|
|
|
|
|
2001-02-12 19:31:01 +00:00
|
|
|
/*
|
|
|
|
* List of available devices, allocated by sane_get_devices, released
|
|
|
|
* by sane_exit()
|
|
|
|
*/
|
|
|
|
static const SANE_Device **devlist = NULL;
|
|
|
|
static int devlist_size = 0, devlist_len = 0;
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
static struct alias *first_alias;
|
1999-08-09 18:06:01 +00:00
|
|
|
static SANE_Auth_Callback auth_callback;
|
|
|
|
static struct backend *first_backend;
|
|
|
|
static const char *op_name[] =
|
|
|
|
{
|
|
|
|
"init", "exit", "get_devices", "open", "close", "get_option_descriptor",
|
|
|
|
"control_option", "get_parameters", "start", "read", "cancel",
|
|
|
|
"set_io_mode", "get_select_fd"
|
|
|
|
};
|
|
|
|
|
|
|
|
static void *
|
|
|
|
op_unsupported (void)
|
|
|
|
{
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(1, "op_unsupported: call to unsupported backend operation\n");
|
1999-08-09 18:06:01 +00:00
|
|
|
return (void *) (long) SANE_STATUS_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static SANE_Status
|
|
|
|
add_backend (const char *name, struct backend **bep)
|
|
|
|
{
|
|
|
|
struct backend *be, *prev;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(1, "add_backend: adding backend %s\n", name);
|
1999-08-09 18:06:01 +00:00
|
|
|
|
2000-10-23 15:03:08 +00:00
|
|
|
if (strcmp(name,"dll") == 0)
|
|
|
|
{
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG( 0, "add_backend: remove the dll-backend from your dll.conf!!!\n");
|
2000-10-23 15:03:08 +00:00
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
1999-08-09 18:06:01 +00:00
|
|
|
for (prev = 0, be = first_backend; be; prev = be, be = be->next)
|
|
|
|
if (strcmp (be->name, name) == 0)
|
|
|
|
{
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(1, "add_backend: %s is already there\n", name);
|
2000-08-12 15:11:46 +00:00
|
|
|
/* move to front so we preserve order that we'd get with
|
1999-08-09 18:06:01 +00:00
|
|
|
dynamic loading: */
|
2000-08-12 15:11:46 +00:00
|
|
|
if (prev)
|
|
|
|
{
|
|
|
|
prev->next = be->next;
|
|
|
|
be->next = first_backend;
|
|
|
|
first_backend = be;
|
|
|
|
}
|
|
|
|
if (bep)
|
|
|
|
*bep = be;
|
|
|
|
return SANE_STATUS_GOOD;
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
be = calloc (1, sizeof (*be));
|
1999-08-09 18:06:01 +00:00
|
|
|
if (!be)
|
|
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
|
|
|
|
be->name = strdup (name);
|
|
|
|
if (!be->name)
|
|
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
be->next = first_backend;
|
|
|
|
first_backend = be;
|
|
|
|
if (bep)
|
|
|
|
*bep = be;
|
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SANE_Status
|
|
|
|
load (struct backend *be)
|
|
|
|
{
|
|
|
|
#ifdef HAVE_DLL
|
|
|
|
int mode = 0;
|
|
|
|
char *funcname, *src, *dir, *path = 0;
|
|
|
|
char libname[PATH_MAX];
|
|
|
|
int i;
|
|
|
|
FILE *fp = 0;
|
|
|
|
|
|
|
|
#if defined(HAVE_DLOPEN)
|
|
|
|
# define PREFIX "libsane-"
|
2001-10-02 20:35:08 +00:00
|
|
|
# ifdef __hpux
|
|
|
|
# define POSTFIX ".sl.%u"
|
|
|
|
# else
|
|
|
|
# define POSTFIX ".so.%u"
|
|
|
|
# endif
|
1999-08-09 18:06:01 +00:00
|
|
|
mode = getenv ("LD_BIND_NOW") ? RTLD_NOW : RTLD_LAZY;
|
|
|
|
#elif defined(HAVE_SHL_LOAD)
|
|
|
|
# define PREFIX "libsane-"
|
|
|
|
# define POSTFIX ".sl.%u"
|
|
|
|
mode = BIND_DEFERRED;
|
|
|
|
#else
|
|
|
|
# error "Tried to compile unsupported DLL."
|
|
|
|
#endif /* HAVE_DLOPEN */
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(1, "load: loading backend %s\n", be->name);
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
/* initialize all ops to "unsupported" so we can "use" the backend
|
|
|
|
even if the stuff later in this function fails */
|
|
|
|
be->loaded = 1;
|
|
|
|
be->handle = 0;
|
|
|
|
for (i = 0; i < NUM_OPS; ++i)
|
|
|
|
be->op[i] = op_unsupported;
|
|
|
|
|
|
|
|
dir = STRINGIFY(LIBDIR);
|
|
|
|
while (dir)
|
|
|
|
{
|
|
|
|
snprintf (libname, sizeof (libname), "%s/"PREFIX"%s"POSTFIX,
|
2000-08-12 15:11:46 +00:00
|
|
|
dir, be->name, V_MAJOR);
|
1999-08-09 18:06:01 +00:00
|
|
|
fp = fopen (libname, "r");
|
|
|
|
if (fp)
|
2000-08-12 15:11:46 +00:00
|
|
|
break;
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
if (!path)
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
path = getenv ("LD_LIBRARY_PATH");
|
|
|
|
if (!path)
|
|
|
|
{
|
|
|
|
path = getenv ("SHLIB_PATH"); /* for HP-UX */
|
|
|
|
if (!path)
|
|
|
|
path = getenv ("LIBPATH"); /* for AIX */
|
|
|
|
}
|
|
|
|
if (!path)
|
|
|
|
break;
|
|
|
|
|
|
|
|
path = strdup (path);
|
|
|
|
src = path;
|
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
dir = strsep (&src, ":");
|
|
|
|
}
|
|
|
|
if (path)
|
|
|
|
free (path);
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
DBG(2, "load: couldn't find %s (%s)\n",
|
2000-08-12 15:11:46 +00:00
|
|
|
libname, strerror (errno));
|
1999-08-09 18:06:01 +00:00
|
|
|
return SANE_STATUS_INVAL;
|
|
|
|
}
|
2001-04-22 20:01:30 +00:00
|
|
|
fclose (fp);
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(2, "load: dlopen()ing `%s'\n", libname);
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_DLOPEN
|
|
|
|
be->handle = dlopen (libname, mode);
|
|
|
|
#elif defined(HAVE_SHL_LOAD)
|
|
|
|
be->handle = (shl_t)shl_load (libname, mode, 0L);
|
|
|
|
#else
|
|
|
|
# error "Tried to compile unsupported DLL."
|
|
|
|
#endif /* HAVE_DLOPEN */
|
|
|
|
if (!be->handle)
|
|
|
|
{
|
2000-03-05 13:57:25 +00:00
|
|
|
#ifdef HAVE_DLOPEN
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(2, "load: dlopen() failed (%s)\n", dlerror());
|
2000-03-05 13:57:25 +00:00
|
|
|
#else
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(2, "load: dlopen() failed (%s)\n", strerror (errno));
|
2000-03-05 13:57:25 +00:00
|
|
|
#endif
|
1999-08-09 18:06:01 +00:00
|
|
|
return SANE_STATUS_INVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* all is dandy---lookup and fill in backend ops: */
|
|
|
|
funcname = alloca (strlen (be->name) + 64);
|
|
|
|
for (i = 0; i < NUM_OPS; ++i)
|
|
|
|
{
|
|
|
|
void *(*op) ();
|
|
|
|
|
|
|
|
sprintf (funcname, "_sane_%s_%s", be->name, op_name[i]);
|
|
|
|
|
|
|
|
/* First try looking up the symbol without a leading underscore. */
|
|
|
|
#ifdef HAVE_DLOPEN
|
|
|
|
op = (void *(*)()) dlsym (be->handle, funcname + 1);
|
|
|
|
#elif defined(HAVE_SHL_LOAD)
|
|
|
|
shl_findsym ((shl_t*)&(be->handle), funcname + 1, TYPE_UNDEFINED, &op);
|
|
|
|
#else
|
|
|
|
# error "Tried to compile unsupported DLL."
|
|
|
|
#endif /* HAVE_DLOPEN */
|
|
|
|
if (op)
|
2000-08-12 15:11:46 +00:00
|
|
|
be->op[i] = op;
|
1999-08-09 18:06:01 +00:00
|
|
|
else
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
/* Try again, with an underscore prepended. */
|
1999-08-09 18:06:01 +00:00
|
|
|
#ifdef HAVE_DLOPEN
|
2000-08-12 15:11:46 +00:00
|
|
|
op = (void *(*)()) dlsym (be->handle, funcname);
|
1999-08-09 18:06:01 +00:00
|
|
|
#elif defined(HAVE_SHL_LOAD)
|
2000-08-12 15:11:46 +00:00
|
|
|
shl_findsym (be->handle, funcname, TYPE_UNDEFINED, &op);
|
1999-08-09 18:06:01 +00:00
|
|
|
#else
|
|
|
|
# error "Tried to compile unsupported DLL."
|
|
|
|
#endif /* HAVE_DLOPEN */
|
2000-08-12 15:11:46 +00:00
|
|
|
if (op)
|
|
|
|
be->op[i] = op;
|
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
if (NULL == op)
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(2, "load: unable to find %s\n", funcname);
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
|
|
|
|
# undef PREFIX
|
|
|
|
# undef POSTFIX
|
|
|
|
#else /* HAVE_DLL */
|
|
|
|
DBG(1, "load: ignoring attempt to load `%s'; compiled without dl support\n",
|
|
|
|
be->name);
|
|
|
|
return SANE_STATUS_UNSUPPORTED;
|
|
|
|
#endif /* HAVE_DLL */
|
|
|
|
}
|
|
|
|
|
|
|
|
static SANE_Status
|
|
|
|
init (struct backend *be)
|
|
|
|
{
|
|
|
|
SANE_Status status;
|
|
|
|
SANE_Int version;
|
|
|
|
|
|
|
|
if (!be->loaded)
|
|
|
|
{
|
|
|
|
status = load (be);
|
|
|
|
if (status != SANE_STATUS_GOOD)
|
2000-08-12 15:11:46 +00:00
|
|
|
return status;
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DBG(3, "init: initializing backend `%s'\n", be->name);
|
|
|
|
|
|
|
|
status = (long) (*be->op[OP_INIT]) (&version, auth_callback);
|
|
|
|
if (status != SANE_STATUS_GOOD)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
if (SANE_VERSION_MAJOR (version) != V_MAJOR)
|
2001-04-13 17:04:51 +00:00
|
|
|
{
|
2001-08-26 10:39:54 +00:00
|
|
|
DBG(1, "init: backend `%s' has a wrong major version (%d instead of %d)\n",
|
2001-04-13 17:04:51 +00:00
|
|
|
be->name, SANE_VERSION_MAJOR (version), V_MAJOR);
|
|
|
|
return SANE_STATUS_INVAL;
|
|
|
|
}
|
|
|
|
DBG(3, "init: backend `%s' is version %d.%d.%d\n", be->name,
|
|
|
|
SANE_VERSION_MAJOR(version), SANE_VERSION_MINOR(version),
|
|
|
|
SANE_VERSION_BUILD(version));
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
be->inited = 1;
|
|
|
|
|
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
add_alias (char *line)
|
|
|
|
{
|
|
|
|
const char *command;
|
|
|
|
enum { CMD_ALIAS, CMD_HIDE } cmd;
|
|
|
|
const char *oldname, *oldend, *newname, *newend;
|
|
|
|
size_t oldlen, newlen;
|
|
|
|
struct alias *alias;
|
|
|
|
|
|
|
|
command = sanei_config_skip_whitespace(line);
|
|
|
|
if( !*command )
|
|
|
|
return;
|
|
|
|
line = strpbrk(command, " \t");
|
|
|
|
if( !line )
|
|
|
|
return;
|
|
|
|
*line++ = '\0';
|
|
|
|
|
|
|
|
if( strcmp(command, "alias") == 0 )
|
|
|
|
cmd = CMD_ALIAS;
|
|
|
|
else
|
|
|
|
if( strcmp(command, "hide") == 0 )
|
|
|
|
cmd = CMD_HIDE;
|
|
|
|
else
|
|
|
|
return;
|
|
|
|
|
|
|
|
newlen = 0;
|
|
|
|
newname = NULL;
|
|
|
|
if( cmd == CMD_ALIAS )
|
|
|
|
{
|
|
|
|
newname = sanei_config_skip_whitespace(line);
|
|
|
|
if( !*newname )
|
2000-08-12 15:11:46 +00:00
|
|
|
return;
|
2000-03-05 13:57:25 +00:00
|
|
|
if( *newname == '\"' )
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
++newname;
|
|
|
|
newend = strchr(newname, '\"');
|
|
|
|
}
|
2000-03-05 13:57:25 +00:00
|
|
|
else
|
2000-08-12 15:11:46 +00:00
|
|
|
newend = strpbrk(newname, " \t");
|
2000-03-05 13:57:25 +00:00
|
|
|
if( !newend )
|
2000-08-12 15:11:46 +00:00
|
|
|
return;
|
2000-03-05 13:57:25 +00:00
|
|
|
|
|
|
|
newlen = newend - newname;
|
|
|
|
line = (char*)(newend+1);
|
|
|
|
}
|
|
|
|
|
|
|
|
oldname = sanei_config_skip_whitespace(line);
|
|
|
|
if( !*oldname )
|
|
|
|
return;
|
|
|
|
oldend = oldname + strcspn(oldname, " \t");
|
|
|
|
|
|
|
|
oldlen = oldend - oldname;
|
|
|
|
|
|
|
|
alias = malloc(sizeof(struct alias));
|
|
|
|
if( alias )
|
|
|
|
{
|
|
|
|
alias->oldname = malloc(oldlen + newlen + 2);
|
|
|
|
if( alias->oldname )
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
strncpy(alias->oldname, oldname, oldlen);
|
|
|
|
alias->oldname[oldlen] = '\0';
|
|
|
|
if( cmd == CMD_ALIAS )
|
|
|
|
{
|
|
|
|
alias->newname = alias->oldname + oldlen + 1;
|
|
|
|
strncpy(alias->newname, newname, newlen);
|
|
|
|
alias->newname[newlen] = '\0';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
alias->newname = NULL;
|
|
|
|
|
|
|
|
alias->next = first_alias;
|
|
|
|
first_alias = alias;
|
|
|
|
return;
|
|
|
|
}
|
2000-03-05 13:57:25 +00:00
|
|
|
free(alias);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-08-09 18:06:01 +00:00
|
|
|
SANE_Status
|
|
|
|
sane_init (SANE_Int * version_code, SANE_Auth_Callback authorize)
|
|
|
|
{
|
|
|
|
char backend_name[PATH_MAX];
|
|
|
|
size_t len;
|
|
|
|
FILE *fp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
DBG_INIT();
|
|
|
|
|
|
|
|
auth_callback = authorize;
|
|
|
|
|
2001-04-16 12:21:41 +00:00
|
|
|
DBG(1, "sane_init: SANE dll backend version %s from %s\n", DLL_VERSION,
|
|
|
|
PACKAGE_VERSION);
|
2001-04-13 17:04:51 +00:00
|
|
|
|
1999-08-09 18:06:01 +00:00
|
|
|
/* chain preloaded backends together: */
|
|
|
|
for (i = 0; i < NELEMS(preloaded_backends); ++i)
|
|
|
|
{
|
|
|
|
if (!preloaded_backends[i].name)
|
2000-08-12 15:11:46 +00:00
|
|
|
continue;
|
1999-08-09 18:06:01 +00:00
|
|
|
preloaded_backends[i].next = first_backend;
|
|
|
|
first_backend = &preloaded_backends[i];
|
|
|
|
}
|
|
|
|
|
2001-04-22 20:01:30 +00:00
|
|
|
/* Return the version number of the sane-backends package to allow
|
|
|
|
the frontend to print them. This is done only for net and dll,
|
|
|
|
because these backends are usually called by the frontend. */
|
1999-08-09 18:06:01 +00:00
|
|
|
if (version_code)
|
2001-04-16 12:21:41 +00:00
|
|
|
*version_code = SANE_VERSION_CODE (SANE_DLL_V_MAJOR, SANE_DLL_V_MINOR,
|
|
|
|
SANE_DLL_V_BUILD);
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
fp = sanei_config_open (DLL_CONFIG_FILE);
|
|
|
|
if (!fp)
|
2000-08-12 15:11:46 +00:00
|
|
|
return SANE_STATUS_GOOD; /* don't insist on config file */
|
1999-08-09 18:06:01 +00:00
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
while (sanei_config_read (backend_name, sizeof (backend_name), fp))
|
1999-08-09 18:06:01 +00:00
|
|
|
{
|
2000-08-12 15:11:46 +00:00
|
|
|
if (backend_name[0] == '#') /* ignore line comments */
|
|
|
|
continue;
|
1999-08-09 18:06:01 +00:00
|
|
|
len = strlen (backend_name);
|
|
|
|
|
|
|
|
if (!len)
|
2000-08-12 15:11:46 +00:00
|
|
|
continue; /* ignore empty lines */
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
add_backend (backend_name, 0);
|
|
|
|
}
|
|
|
|
fclose (fp);
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
fp = sanei_config_open (DLL_ALIASES_FILE);
|
|
|
|
if (!fp)
|
2000-08-12 15:11:46 +00:00
|
|
|
return SANE_STATUS_GOOD; /* don't insist on aliases file */
|
2000-03-05 13:57:25 +00:00
|
|
|
|
2000-08-12 15:11:46 +00:00
|
|
|
while (sanei_config_read (backend_name, sizeof (backend_name), fp))
|
2000-03-05 13:57:25 +00:00
|
|
|
{
|
2000-08-12 15:11:46 +00:00
|
|
|
if (backend_name[0] == '#') /* ignore line comments */
|
|
|
|
continue;
|
2000-03-05 13:57:25 +00:00
|
|
|
len = strlen (backend_name);
|
|
|
|
|
|
|
|
if (!len)
|
2000-08-12 15:11:46 +00:00
|
|
|
continue; /* ignore empty lines */
|
2000-03-05 13:57:25 +00:00
|
|
|
|
|
|
|
add_alias (backend_name);
|
|
|
|
}
|
|
|
|
fclose (fp);
|
1999-08-09 18:06:01 +00:00
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sane_exit (void)
|
|
|
|
{
|
|
|
|
struct backend *be, *next;
|
2000-03-05 13:57:25 +00:00
|
|
|
struct alias *alias;
|
1999-08-09 18:06:01 +00:00
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(1, "sane_exit: exiting\n");
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
for (be = first_backend; be; be = next)
|
|
|
|
{
|
|
|
|
next = be->next;
|
|
|
|
if (be->loaded)
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(2, "sane_exit: calling backend `%s's exit function\n", be->name);
|
2000-08-12 15:11:46 +00:00
|
|
|
(*be->op[OP_EXIT]) ();
|
1999-08-09 18:06:01 +00:00
|
|
|
#ifdef HAVE_DLL
|
|
|
|
|
|
|
|
#ifdef HAVE_DLOPEN
|
2000-08-12 15:11:46 +00:00
|
|
|
if (be->handle)
|
|
|
|
dlclose (be->handle);
|
1999-08-09 18:06:01 +00:00
|
|
|
#elif defined(HAVE_SHL_LOAD)
|
2000-08-12 15:11:46 +00:00
|
|
|
if (be->handle)
|
|
|
|
shl_unload(be->handle);
|
1999-08-09 18:06:01 +00:00
|
|
|
#else
|
|
|
|
# error "Tried to compile unsupported DLL."
|
|
|
|
#endif /* HAVE_DLOPEN */
|
|
|
|
|
|
|
|
#endif /* HAVE_DLL */
|
2000-08-12 15:11:46 +00:00
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
if (!be->permanent)
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
if (be->name)
|
|
|
|
free ((void *) be->name);
|
|
|
|
free (be);
|
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
first_backend = 0;
|
2000-03-05 13:57:25 +00:00
|
|
|
|
|
|
|
while( (alias = first_alias) != NULL )
|
|
|
|
{
|
|
|
|
first_alias = first_alias->next;
|
|
|
|
free(alias->oldname);
|
|
|
|
free(alias);
|
|
|
|
}
|
2001-02-12 19:31:01 +00:00
|
|
|
|
|
|
|
if (NULL != devlist)
|
|
|
|
{ /* Release memory allocated by sane_get_devices(). */
|
|
|
|
int i = 0;
|
|
|
|
while (devlist[i])
|
|
|
|
free(devlist[i++]);
|
|
|
|
free(devlist);
|
|
|
|
|
|
|
|
devlist = NULL;
|
|
|
|
devlist_size = 0;
|
|
|
|
devlist_len = 0;
|
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Note that a call to get_devices() implies that we'll have to load
|
|
|
|
all backends. To avoid this, you can call sane_open() directly
|
|
|
|
(assuming you know the name of the backend/device). This is
|
|
|
|
appropriate for the command-line interface of SANE, for example.
|
|
|
|
*/
|
|
|
|
SANE_Status
|
|
|
|
sane_get_devices (const SANE_Device *** device_list, SANE_Bool local_only)
|
|
|
|
{
|
|
|
|
const SANE_Device **be_list;
|
|
|
|
struct backend *be;
|
|
|
|
SANE_Status status;
|
|
|
|
char *full_name;
|
|
|
|
int i, num_devs;
|
|
|
|
size_t len;
|
2000-08-12 15:11:46 +00:00
|
|
|
#define ASSERT_SPACE(n) \
|
|
|
|
{ \
|
|
|
|
if (devlist_len + (n) > devlist_size) \
|
|
|
|
{ \
|
|
|
|
devlist_size += (n) + 15; \
|
|
|
|
if (devlist) \
|
|
|
|
devlist = realloc (devlist, devlist_size * sizeof (devlist[0])); \
|
|
|
|
else \
|
|
|
|
devlist = malloc (devlist_size * sizeof (devlist[0])); \
|
|
|
|
if (!devlist) \
|
|
|
|
return SANE_STATUS_NO_MEM; \
|
|
|
|
} \
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (devlist)
|
|
|
|
for (i = 0; i < devlist_len; ++i)
|
|
|
|
free ((void *) devlist[i]);
|
|
|
|
devlist_len = 0;
|
|
|
|
|
|
|
|
for (be = first_backend; be; be = be->next)
|
|
|
|
{
|
|
|
|
if (!be->inited)
|
2000-08-12 15:11:46 +00:00
|
|
|
if (init (be) != SANE_STATUS_GOOD)
|
|
|
|
continue;
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
status = (long) (*be->op[OP_GET_DEVS]) (&be_list, local_only);
|
|
|
|
if (status != SANE_STATUS_GOOD || !be_list)
|
2000-08-12 15:11:46 +00:00
|
|
|
continue;
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
/* count the number of devices for this backend: */
|
|
|
|
for (num_devs = 0; be_list[num_devs]; ++num_devs);
|
|
|
|
|
|
|
|
ASSERT_SPACE (num_devs);
|
|
|
|
|
|
|
|
for (i = 0; i < num_devs; ++i)
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
SANE_Device *dev;
|
|
|
|
char *mem;
|
|
|
|
struct alias *alias;
|
|
|
|
|
|
|
|
for(alias = first_alias; alias != NULL; alias = alias->next)
|
|
|
|
{
|
|
|
|
len = strlen(be->name);
|
|
|
|
if( strlen(alias->oldname) <= len )
|
|
|
|
continue;
|
|
|
|
if( strncmp(alias->oldname, be->name, len) == 0
|
|
|
|
&& alias->oldname[len] == ':'
|
|
|
|
&& strcmp(&alias->oldname[len+1], be_list[i]->name) == 0 )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( alias )
|
|
|
|
{
|
|
|
|
if( !alias->newname ) /* hidden device */
|
|
|
|
continue;
|
|
|
|
|
|
|
|
len = strlen(alias->newname);
|
|
|
|
mem = malloc(sizeof(*dev) + len + 1);
|
|
|
|
if( !mem )
|
|
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
|
|
|
|
full_name = mem + sizeof(*dev);
|
|
|
|
strcpy(full_name, alias->newname);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* create a new device entry with a device name that is the
|
|
|
|
sum of the backend name a colon and the backend's device
|
|
|
|
name: */
|
|
|
|
len = strlen (be->name) + 1 + strlen (be_list[i]->name);
|
|
|
|
mem = malloc (sizeof (*dev) + len + 1);
|
|
|
|
if (!mem)
|
|
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
|
|
|
|
full_name = mem + sizeof (*dev);
|
|
|
|
strcpy (full_name, be->name);
|
|
|
|
strcat (full_name, ":");
|
|
|
|
strcat (full_name, be_list[i]->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
dev = (SANE_Device *) mem;
|
|
|
|
dev->name = full_name;
|
|
|
|
dev->vendor = be_list[i]->vendor;
|
|
|
|
dev->model = be_list[i]->model;
|
|
|
|
dev->type = be_list[i]->type;
|
|
|
|
|
|
|
|
devlist[devlist_len++] = dev;
|
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* terminate device list with NULL entry: */
|
|
|
|
ASSERT_SPACE (1);
|
|
|
|
devlist[devlist_len++] = 0;
|
|
|
|
|
|
|
|
*device_list = devlist;
|
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_open (SANE_String_Const full_name, SANE_Handle * meta_handle)
|
|
|
|
{
|
|
|
|
const char *be_name, *dev_name;
|
|
|
|
struct meta_scanner *s;
|
|
|
|
SANE_Handle *handle;
|
|
|
|
struct backend *be;
|
|
|
|
SANE_Status status;
|
2000-03-05 13:57:25 +00:00
|
|
|
struct alias *alias;
|
|
|
|
|
|
|
|
for( alias = first_alias; alias != NULL; alias = alias->next )
|
|
|
|
{
|
|
|
|
if( !alias->newname )
|
2000-08-12 15:11:46 +00:00
|
|
|
continue;
|
2000-03-05 13:57:25 +00:00
|
|
|
if( strcmp(alias->newname, full_name) == 0 )
|
2000-08-12 15:11:46 +00:00
|
|
|
{
|
|
|
|
full_name = alias->oldname;
|
|
|
|
break;
|
|
|
|
}
|
2000-03-05 13:57:25 +00:00
|
|
|
}
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
dev_name = strchr (full_name, ':');
|
|
|
|
if (dev_name)
|
|
|
|
{
|
|
|
|
#ifdef strndupa
|
|
|
|
be_name = strndupa (full_name, dev_name - full_name);
|
|
|
|
#else
|
|
|
|
char *tmp;
|
|
|
|
|
|
|
|
tmp = alloca (dev_name - full_name + 1);
|
|
|
|
memcpy (tmp, full_name, dev_name - full_name);
|
|
|
|
tmp[dev_name - full_name] = '\0';
|
|
|
|
be_name = tmp;
|
|
|
|
#endif
|
2000-08-12 15:11:46 +00:00
|
|
|
++dev_name; /* skip colon */
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* if no colon interpret full_name as the backend name; an empty
|
|
|
|
backend device name will cause us to open the first device of
|
|
|
|
that backend. */
|
|
|
|
be_name = full_name;
|
|
|
|
dev_name = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!be_name[0])
|
|
|
|
be = first_backend;
|
|
|
|
else
|
|
|
|
for (be = first_backend; be; be = be->next)
|
|
|
|
if (strcmp (be->name, be_name) == 0)
|
2000-08-12 15:11:46 +00:00
|
|
|
break;
|
1999-08-09 18:06:01 +00:00
|
|
|
|
|
|
|
if (!be)
|
|
|
|
{
|
|
|
|
status = add_backend (be_name, &be);
|
|
|
|
if (status != SANE_STATUS_GOOD)
|
2000-08-12 15:11:46 +00:00
|
|
|
return status;
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!be->inited)
|
|
|
|
{
|
|
|
|
status = init (be);
|
|
|
|
if (status != SANE_STATUS_GOOD)
|
2000-08-12 15:11:46 +00:00
|
|
|
return status;
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
status = (long) (*be->op[OP_OPEN]) (dev_name, &handle);
|
|
|
|
if (status != SANE_STATUS_GOOD)
|
|
|
|
return status;
|
|
|
|
|
2000-03-05 13:57:25 +00:00
|
|
|
s = calloc (1, sizeof (*s));
|
1999-08-09 18:06:01 +00:00
|
|
|
if (!s)
|
|
|
|
return SANE_STATUS_NO_MEM;
|
|
|
|
|
|
|
|
s->be = be;
|
|
|
|
s->handle = handle;
|
|
|
|
*meta_handle = s;
|
|
|
|
|
|
|
|
return SANE_STATUS_GOOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sane_close (SANE_Handle handle)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_close(handle=%p)\n", handle);
|
1999-08-09 18:06:01 +00:00
|
|
|
(*s->be->op[OP_CLOSE]) (s->handle);
|
|
|
|
free (s);
|
|
|
|
}
|
|
|
|
|
|
|
|
const SANE_Option_Descriptor *
|
|
|
|
sane_get_option_descriptor (SANE_Handle handle, SANE_Int option)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_get_option_descriptor(handle=%p,option=%d)\n", handle, option);
|
1999-08-09 18:06:01 +00:00
|
|
|
return (*s->be->op[OP_GET_OPTION_DESC]) (s->handle, option);
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_control_option (SANE_Handle handle, SANE_Int option,
|
2000-08-12 15:11:46 +00:00
|
|
|
SANE_Action action, void *value, SANE_Word * info)
|
1999-08-09 18:06:01 +00:00
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_control_option(handle=%p,option=%d,action=%d,value=%p,info=%p)\n",
|
1999-08-09 18:06:01 +00:00
|
|
|
handle, option, action, value, info);
|
|
|
|
return (long) (*s->be->op[OP_CTL_OPTION]) (s->handle, option, action,
|
2000-08-12 15:11:46 +00:00
|
|
|
value, info);
|
1999-08-09 18:06:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_get_parameters (SANE_Handle handle, SANE_Parameters * params)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_get_parameters(handle=%p,params=%p)\n", handle, params);
|
1999-08-09 18:06:01 +00:00
|
|
|
return (long) (*s->be->op[OP_GET_PARAMS]) (s->handle, params);
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_start (SANE_Handle handle)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_start(handle=%p)\n", handle);
|
1999-08-09 18:06:01 +00:00
|
|
|
return (long) (*s->be->op[OP_START]) (s->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_read (SANE_Handle handle, SANE_Byte * data, SANE_Int max_length,
|
2000-08-12 15:11:46 +00:00
|
|
|
SANE_Int * length)
|
1999-08-09 18:06:01 +00:00
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_read(handle=%p,data=%p,maxlen=%d,lenp=%p)\n",
|
1999-08-09 18:06:01 +00:00
|
|
|
handle, data, max_length, length);
|
|
|
|
return (long) (*s->be->op[OP_READ]) (s->handle, data, max_length, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sane_cancel (SANE_Handle handle)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_cancel(handle=%p)\n", handle);
|
1999-08-09 18:06:01 +00:00
|
|
|
(*s->be->op[OP_CANCEL]) (s->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_set_io_mode (SANE_Handle handle, SANE_Bool non_blocking)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_set_io_mode(handle=%p,nonblocking=%d)\n", handle, non_blocking);
|
1999-08-09 18:06:01 +00:00
|
|
|
return (long) (*s->be->op[OP_SET_IO_MODE]) (s->handle, non_blocking);
|
|
|
|
}
|
|
|
|
|
|
|
|
SANE_Status
|
|
|
|
sane_get_select_fd (SANE_Handle handle, SANE_Int * fd)
|
|
|
|
{
|
|
|
|
struct meta_scanner *s = handle;
|
|
|
|
|
2001-04-13 17:04:51 +00:00
|
|
|
DBG(3, "sane_get_select_fd(handle=%p,fdp=%p)\n", handle, fd);
|
1999-08-09 18:06:01 +00:00
|
|
|
return (long) (*s->be->op[OP_GET_SELECT_FD]) (s->handle, fd);
|
|
|
|
}
|