aboutsummaryrefslogtreecommitdiff
path: root/backend/backend.c
diff options
context:
space:
mode:
Diffstat (limited to 'backend/backend.c')
-rw-r--r--backend/backend.c600
1 files changed, 600 insertions, 0 deletions
diff --git a/backend/backend.c b/backend/backend.c
new file mode 100644
index 0000000..c9e6cf0
--- /dev/null
+++ b/backend/backend.c
@@ -0,0 +1,600 @@
+/* backend.c -- implements the SANE epkowa backend
+ * Copyright (C) 2008--2009 SEIKO EPSON CORPORATION
+ *
+ * License: GPLv2+|iscan
+ * Authors: AVASYS CORPORATION
+ *
+ * This file is part of the SANE backend distributed with Image Scan!
+ *
+ * Image Scan!'s SANE backend 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 ought to have received a copy of the GNU General Public License
+ * along with this package. If not, see <http://www.gnu.org/licenses/>.
+ *
+ *
+ * Linking Image Scan!'s SANE backend statically or dynamically with
+ * other modules is making a combined work based on this SANE backend.
+ * Thus, the terms and conditions of the GNU General Public License
+ * cover the whole combination.
+ *
+ * As a special exception, the copyright holders of Image Scan!'s SANE
+ * backend give you permission to link Image Scan!'s SANE backend with
+ * SANE frontends that communicate with Image Scan!'s SANE backend
+ * solely through the SANE Application Programming Interface,
+ * regardless of the license terms of these SANE frontends, and to
+ * copy and distribute the resulting combined work under terms of your
+ * choice, provided that every copy of the combined work is
+ * accompanied by a complete copy of the source code of Image Scan!'s
+ * SANE backend (the version of Image Scan!'s SANE backend used to
+ * produce the combined work), being distributed under the terms of
+ * the GNU General Public License plus this exception. An independent
+ * module is a module which is not derived from or based on Image
+ * Scan!'s SANE backend.
+ *
+ * As a special exception, the copyright holders of Image Scan!'s SANE
+ * backend give you permission to link Image Scan!'s SANE backend with
+ * independent modules that communicate with Image Scan!'s SANE
+ * backend solely through the "Interpreter" interface, regardless of
+ * the license terms of these independent modules, and to copy and
+ * distribute the resulting combined work under terms of your choice,
+ * provided that every copy of the combined work is accompanied by a
+ * complete copy of the source code of Image Scan!'s SANE backend (the
+ * version of Image Scan!'s SANE backend used to produce the combined
+ * work), being distributed under the terms of the GNU General Public
+ * License plus this exception. An independent module is a module
+ * which is not derived from or based on Image Scan!'s SANE backend.
+ *
+ * Note that people who make modified versions of Image Scan!'s SANE
+ * backend are not obligated to grant special exceptions for their
+ * modified versions; it is their choice whether to do so. The GNU
+ * General Public License gives permission to release a modified
+ * version without this exception; this exception also makes it
+ * possible to release a modified version which carries forward this
+ * exception.
+ */
+
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+
+#define BACKEND_CREATE_FORWARDERS
+#include "backend.h"
+
+#define BACKEND_BUILD 213
+#define BACKEND_SOURCE PACKAGE_STRING
+
+#include <errno.h>
+#include <ltdl.h>
+#include <string.h>
+
+#include "cfg-obj.h"
+#include "dip-obj.h"
+#include "net-obj.h"
+#include "model-info.h"
+#include "utils.h"
+
+/* Deprecated includes */
+#include "include/sane/sanei_usb.h"
+#include "epkowa.h"
+
+/* Deprecated API calls */
+extern SANE_Status epkowa_open (const char *, SANE_Handle *, const void *);
+
+
+typedef struct
+{
+ void *cfg;
+ void *net;
+ void *dip;
+ list *sane_dev;
+ void **dev_list;
+ void *model_info_cache;
+
+} backend_type;
+
+static backend_type *be = NULL;
+
+
+static void be_sane_dev_dtor (void *);
+
+
+/*! \defgroup SANE_API SANE API Entry Points
+ *
+ * The SANE API entry points make up the \e full API available to the
+ * SANE frontend application programmer. Users of this API should be
+ * careful \e never to assume \e anything about a backend's behaviour
+ * beyond what is required by the SANE standard. The standard can be
+ * retrieved via http://www.sane-project.org/docs.html.
+ *
+ * Whatever documentation may be provided here serves to document the
+ * implementation, if anything. In case of discrepancy with the SANE
+ * specification, the SANE specification is correct.
+ *
+ * @{
+ */
+
+
+/*! \brief Prepares the backend for use by a SANE frontend.
+ *
+ * \remarks
+ * This function \e must be called before any other SANE API entry is
+ * called. It is the only SANE function that may be called after the
+ * sane_exit() function has been called.
+ *
+ * \todo
+ * Deal properly with the \c SANE_Auth_Callback.
+ * It is completely ignored in the current implementation.
+ */
+SANE_Status
+sane_init (SANE_Int *version_code, SANE_Auth_Callback authorize)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+
+ if (be)
+ {
+ log_call ("(%p, %p)", version_code, authorize);
+ err_minor ("backend already initialised");
+ return status;
+ }
+
+ msg_init ();
+ log_call ("(%p, %p)", version_code, authorize);
+ log_info ("%s", BACKEND_SOURCE);
+ log_info ("version %d.%d.%d", SANE_MAJOR, SANE_MINOR, BACKEND_BUILD);
+
+ if (version_code)
+ {
+ *version_code = SANE_VERSION_CODE (SANE_MAJOR, SANE_MINOR,
+ BACKEND_BUILD);
+ }
+
+ if (authorize)
+ {
+ err_minor ("authorisation not supported");
+ }
+
+ be = t_calloc (1, backend_type);
+ if (!be)
+ {
+ return SANE_STATUS_NO_MEM;
+ }
+
+ /* Needs to be done _before_ cfg_init() because that refers to the
+ * model_info_cache somewhere deep down in its bowels.
+ */
+ be->model_info_cache = model_info_cache_init (MODELDATADIR, &status);
+ if (!be->model_info_cache)
+ {
+ sane_exit ();
+ return status;
+ }
+
+ be->cfg = cfg_init (PKGDATADIR, &status);
+ if (!be->cfg)
+ {
+ sane_exit ();
+ return status;
+ }
+
+ if (cfg_has (be->cfg, CFG_KEY_NET))
+ {
+ be->net = net_init (PKGLIBDIR, &status);
+ if (!be->net)
+ {
+ if (SANE_STATUS_GOOD != status)
+ err_fatal ("%s", sane_strstatus (status));
+ err_major ("disabling network device support");
+ cfg_set (be->cfg, CFG_KEY_NET, false);
+ status = SANE_STATUS_GOOD;
+ }
+ }
+ if (cfg_has (be->cfg, CFG_KEY_PIO))
+ {
+ /* place holder */
+ }
+ if (cfg_has (be->cfg, CFG_KEY_SCSI))
+ {
+ /* place holder */
+ }
+ if (cfg_has (be->cfg, CFG_KEY_USB))
+ {
+ sanei_usb_init ();
+ }
+
+ if (cfg_has (be->cfg, CFG_KEY_INTERPRETER))
+ {
+ if (0 != lt_dlinit ())
+ {
+ err_fatal ("%s", lt_dlerror ());
+ err_major ("disabling interpreter support");
+ cfg_set (be->cfg, CFG_KEY_INTERPRETER, false);
+ }
+ }
+
+ be->dip = dip_init (PKGLIBDIR, &status);
+ if (!be->dip)
+ {
+ sane_exit ();
+ return status;
+ }
+
+ return status;
+}
+
+
+/*! \brief Releases all resources held by the backend.
+ *
+ * \remarks
+ * Applications \e must call this function to terminate use of the
+ * backend. After it has been called, sane_init() has to be called
+ * before other SANE API can be used. The function needs to close
+ * any open handles.
+ *
+ * \remarks
+ * The implementation must be able to deal properly with a partially
+ * initialised backend so that sane_init() can use this function for
+ * its error recovery.
+ */
+void
+sane_exit (void)
+{
+ log_call ("()");
+
+ if (!be)
+ {
+ msg_init ();
+ err_minor ("backend is not initialized");
+ return;
+ }
+
+ be->dip = dip_exit (be->dip);
+
+ if (cfg_has (be->cfg, CFG_KEY_INTERPRETER))
+ {
+ lt_dlexit ();
+ }
+
+ if (cfg_has (be->cfg, CFG_KEY_USB))
+ {
+ /* sanei_usb_exit, if only there was one! */
+ }
+ if (cfg_has (be->cfg, CFG_KEY_SCSI))
+ {
+ /* place holder */
+ }
+ if (cfg_has (be->cfg, CFG_KEY_PIO))
+ {
+ /* place holder */
+ }
+ if (be->net)
+ {
+ be->net = net_exit (be->net);
+ }
+
+ be->cfg = cfg_exit (be->cfg);
+ delete (be->dev_list);
+ list_destroy (be->sane_dev, be_sane_dev_dtor);
+
+ be->model_info_cache = model_info_cache_exit (be->model_info_cache);
+
+ delete (be);
+}
+
+
+/*! \brief Creates a list of devices available through the backend.
+ *
+ * \remarks
+ * The returned \a device_list \e must remain unchanged and valid
+ * until this function is called again or sane_exit() is called.
+ *
+ * \remarks
+ * Applications are \e not required to call this function before
+ * they call sane_open().
+ */
+SANE_Status
+sane_get_devices (const SANE_Device ***device_list, SANE_Bool local_only)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ list *sane_dev = NULL;
+
+ log_call ("(%p, %d)", device_list, local_only);
+
+ if (!be)
+ {
+ msg_init ();
+ err_fatal ("backend is not initialized");
+ return SANE_STATUS_ACCESS_DENIED;
+ }
+
+ if (!device_list)
+ {
+ err_fatal ("%s", strerror (EINVAL));
+ return SANE_STATUS_INVAL;
+ }
+
+ sane_dev = list_create ();
+ if (sane_dev)
+ {
+ if (!local_only && cfg_has (be->cfg, CFG_KEY_NET))
+ {
+ cfg_find (be->cfg, CFG_KEY_NET, sane_dev);
+ }
+ if (cfg_has (be->cfg, CFG_KEY_PIO))
+ {
+ cfg_find (be->cfg, CFG_KEY_PIO, sane_dev);
+ }
+ if (cfg_has (be->cfg, CFG_KEY_SCSI))
+ {
+ cfg_find (be->cfg, CFG_KEY_SCSI, sane_dev);
+ }
+ if (cfg_has (be->cfg, CFG_KEY_USB))
+ {
+ cfg_find (be->cfg, CFG_KEY_USB, sane_dev);
+ }
+
+ if (cfg_has (be->cfg, CFG_KEY_INTERPRETER))
+ {
+ cfg_find (be->cfg, CFG_KEY_INTERPRETER, sane_dev);
+ }
+
+ if (be->sane_dev)
+ {
+ delete (be->dev_list);
+ list_destroy (be->sane_dev, be_sane_dev_dtor);
+ }
+ be->sane_dev = sane_dev;
+ }
+
+ be->dev_list = list_normalize (be->sane_dev);
+ *device_list = (const SANE_Device **) be->dev_list;
+ if (!*device_list)
+ {
+ status = SANE_STATUS_NO_MEM;
+ }
+
+ return status;
+}
+
+
+/*! \brief Establishes a connection to a named device.
+ *
+ * \remarks
+ * Applications are allowed to call this function directly, without a
+ * call to sane_get_devices() first. An empty string may be used for
+ * the \a device_name to request the first available device.
+ *
+ * \todo Register the handle with \c be before \c SANE_STATUS_GOOD is
+ * returned so we can check in the other API entries whether we
+ * got a known handle passed in as well as call sane_close() on
+ * all open handles in sane_exit().
+ */
+SANE_Status
+sane_open (SANE_String_Const device_name, SANE_Handle *handle)
+{
+ const SANE_Device *sane_dev = NULL;
+
+ log_call ("(%s, %p)", device_name, handle);
+
+ if (!be)
+ {
+ msg_init ();
+ err_fatal ("backend is not initialized");
+ return SANE_STATUS_ACCESS_DENIED;
+ }
+
+ if (!handle)
+ {
+ err_fatal ("%s", strerror (EINVAL));
+ return SANE_STATUS_INVAL;
+ }
+
+ if (!device_name)
+ {
+ /* The SANE API specification explicitly talks about a zero
+ * length string. There is no mention about what should be
+ * done in the case where there is NO string at all.
+ * We degrade gracefully.
+ */
+ err_minor ("assuming frontend meant to pass an empty string");
+ }
+
+ if (!be->sane_dev)
+ { /* FIXME: does more than necessary */
+ const SANE_Device **dev = NULL;
+ sane_get_devices (&dev, false);
+ }
+
+ if (0 == list_size (be->sane_dev))
+ {
+ err_major ("no supported devices available");
+ return SANE_STATUS_ACCESS_DENIED;
+ }
+
+ if (!device_name || 0 == strlen (device_name))
+ {
+ sane_dev = be->sane_dev->head->data;
+ }
+ else
+ {
+ list_reset (be->sane_dev);
+ while ((sane_dev = list_next (be->sane_dev))
+ && 0 != strcmp_c (sane_dev->name, device_name))
+ {
+ /* nothing to do, condition does all the processing */
+ }
+ }
+
+ if (!sane_dev)
+ {
+ err_major ("no such device");
+ return SANE_STATUS_INVAL;
+ }
+
+ return epkowa_open (sane_dev->name, handle, be->dip);
+}
+
+//! Obtains the current scan parameters for a device
+/*! \remarks
+ * The parameters are only guaranteed to be accurate between a call
+ * to sane_start() and the completion of that request. Outside of
+ * that scope the parameters are a best effort only and the backend
+ * is at liberty to change them.
+ */
+SANE_Status
+sane_get_parameters (SANE_Handle handle, SANE_Parameters *parameters)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ Epson_Scanner *s;
+
+ log_call ("(%p, %p)", handle, parameters);
+
+ if (!handle || !parameters)
+ {
+ err_fatal ("%s", strerror (EINVAL));
+ return SANE_STATUS_INVAL;
+ }
+
+ s = (Epson_Scanner *) handle;
+
+ if (s->src->transfer_started && !s->src->transfer_stopped)
+ {
+ static const char *const color_space[] = {
+ "GRAY", "RGB", "RED", "GREEN", "BLUE"
+ };
+
+ const SANE_Parameters *p = &s->src->ctx;
+
+ log_info
+ ("Scan area : %.2f x %.2f [mm^2]",
+ SANE_UNFIX (s->val[OPT_BR_X].w) - SANE_UNFIX (s->val[OPT_TL_X].w),
+ SANE_UNFIX (s->val[OPT_BR_Y].w) - SANE_UNFIX (s->val[OPT_TL_Y].w));
+ log_info
+ ("Offset : (%.2f, %.2f) [mm]",
+ SANE_UNFIX (s->val[OPT_TL_X].w), SANE_UNFIX (s->val[OPT_TL_Y].w));
+
+ log_info ("Color space : %s-%d", color_space[p->format], p->depth);
+ log_info ("Image size : %d x %d [pixels^2] (%.2f x %.2f [mm^2])",
+ p->pixels_per_line, p->lines,
+ p->pixels_per_line * MM_PER_INCH / s->val[OPT_X_RESOLUTION].w,
+ p->lines * MM_PER_INCH / s->val[OPT_Y_RESOLUTION].w);
+ log_info ("X Resolution: %d [dpi]", s->val[OPT_X_RESOLUTION].w);
+ log_info ("Y Resolution: %d [dpi]", s->val[OPT_Y_RESOLUTION].w);
+
+ memcpy (parameters, p, sizeof (*p));
+ }
+ else
+ {
+ status = estimate_parameters (s, parameters);
+ }
+
+ return status;
+}
+
+/*! \brief Acquires up to \a max_length bytes of new image data.
+ *
+ * \remarks
+ * The \a length is guaranteed to be zero in case of an unsuccessful
+ * request.
+ *
+ * \remarks
+ * The implementation allows for \c NULL as a \a buffer value. This
+ * caters to a frontends that do not prepare buffer space when they
+ * expect a \c SANE_STATUS_EOF return value.
+ */
+SANE_Status
+sane_read (SANE_Handle handle, SANE_Byte *buffer,
+ SANE_Int max_length, SANE_Int *length)
+{
+ SANE_Status status = SANE_STATUS_GOOD;
+ Epson_Scanner *s;
+
+ log_call ("(%p, %p, %i, %p)", handle, buffer, max_length, length);
+
+ if (length) *length = 0;
+
+ if (!handle)
+ {
+ err_fatal ("%s", strerror (EINVAL));
+ return SANE_STATUS_INVAL;
+ }
+
+ s = (Epson_Scanner *) handle;
+
+ require (s->src == &s->raw || s->src == &s->img);
+
+ if (s->src == &s->raw)
+ {
+ status = fetch_image_data (s, buffer, max_length, length);
+ }
+ else if (s->src == &s->img)
+ {
+ /**/ if (!s->img.ptr)
+ {
+ err_major ("%s", strerror (ENOMEM));
+ status = SANE_STATUS_NO_MEM;
+ }
+ else if (s->img.ptr == s->img.end)
+ {
+ status = SANE_STATUS_EOF;
+ }
+ else if (s->img.cancel_requested)
+ {
+ s->img.transfer_stopped = true;
+ status = SANE_STATUS_CANCELLED;
+ }
+ else if (buffer && 0 < max_length)
+ {
+ SANE_Int len = s->img.end - s->img.ptr;
+
+ if (len > max_length) len = max_length;
+ memcpy (buffer, s->img.ptr, len);
+ s->img.ptr += len;
+ if (length) *length = len;
+ }
+ else
+ {
+ status = SANE_STATUS_NO_MEM;
+ }
+ }
+
+ if (SANE_STATUS_EOF == status)
+ {
+ s->src->transfer_stopped = true;
+ }
+
+ return status;
+}
+
+/*!
+ @}
+ */
+
+
+/*! Releases the resources held by a SANE_Device.
+
+ This function is primarily useful to maintain the \c sane_dev
+ member of a backend_type object.
+ */
+static void
+be_sane_dev_dtor (void *p)
+{
+ SANE_Device *sd = (SANE_Device *) p;
+ if (!sd) return;
+
+ const_delete (sd->name , SANE_String);
+ const_delete (sd->vendor, SANE_String);
+ const_delete (sd->model , SANE_String);
+ const_delete (sd->type , SANE_String);
+
+ delete (sd);
+}