/* channel_net.c -- network device communication channel * 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 . * * * 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. */ /*! \file * \brief Implements a network device communication channel. * \todo Deal correctly with \c SIGPIPE. */ #ifdef HAVE_CONFIG_H #include #endif #include "channel.h" #include #include #include "ipc.h" #include "net-obj.h" #include "utils.h" #include "hw-data.h" static void channel_net_open (channel *, SANE_Status *); static void channel_net_close (channel *, SANE_Status *); static ssize_t channel_net_send (channel *, const void *, size_t, SANE_Status *); static ssize_t channel_net_recv (channel *, void *, size_t, SANE_Status *); channel * channel_net_ctor (channel *self, const char *dev_name, SANE_Status *status) { log_call ("(%p, '%s', %p)", self, dev_name, status); if (status) *status = SANE_STATUS_GOOD; require (self && dev_name); require (0 == strncmp_c (dev_name, "net:", strlen ("net:"))); self->name = strdup (dev_name); if (!self->name) { if (status) *status = SANE_STATUS_NO_MEM; return self->dtor (self); } self->open = channel_net_open; self->close = channel_net_close; self->send = channel_net_send; self->recv = channel_net_recv; return self; } /*! Each call must consist of a complete handshake step. * Cannot send in parts. */ static ssize_t channel_net_send (channel *self, const void *buffer, size_t size, SANE_Status *status) { ssize_t n = 0; if (0 > self->fd) { if (status) *status = SANE_STATUS_IO_ERROR; return -1; } if (status) *status = SANE_STATUS_GOOD; require (self && buffer); require (0 < self->id); n = ipc_send (self->fd, self->id, TYPE_ESC, size, buffer); if (n != size) { if (status) *status = SANE_STATUS_IO_ERROR; } return n; } /*! Each call must consist of a complete handshake step. * Cannot receive in parts. */ static ssize_t channel_net_recv (channel *self, void *buffer, size_t size, SANE_Status *status) { char* rbuf = NULL; uint16_t id = 0; uint8_t ipc_status = STATUS_OK; ssize_t n = 0; if (0 > self->fd) { if (status) *status = SANE_STATUS_IO_ERROR; return -1; } if (status) *status = SANE_STATUS_GOOD; require (self && buffer); require (0 < self->id); n = ipc_recv (self->fd, &id, &ipc_status, (void**)&rbuf); if (n != size) err_major ("expected %zd bytes, received %zd bytes", size, n); if (!rbuf || id != self->id || STATUS_OK != ipc_status || n != size) { if (status) *status = SANE_STATUS_IO_ERROR; delete (rbuf); return -1; } memcpy (buffer, rbuf, n); delete (rbuf); return n; } static void channel_net_open (channel *self, SANE_Status *status) { void* net = NULL; ssize_t n = 0; uint8_t ipc_status = STATUS_OK; char* scanner = self->name + strlen ("net:"); if (status) *status = SANE_STATUS_GOOD; net = net_init ("", NULL); if (net) self->fd = net_get_sock (net); if (!net || 0 > self->fd) { if (status) *status = SANE_STATUS_IO_ERROR; return; } n = ipc_send (self->fd, 0, TYPE_OPEN, strlen (scanner), scanner); if (n != strlen (scanner)) { self->fd = -1; if (status) *status = SANE_STATUS_IO_ERROR; return; } n = ipc_recv (self->fd, &self->id, &ipc_status, NULL); if (STATUS_OK != ipc_status || 0 != n) { self->id = 0; self->fd = -1; if (status) *status = SANE_STATUS_IO_ERROR; return; } log_info ("Opened network scanner at: %s", scanner); } static void channel_net_close (channel *self, SANE_Status *status) { ssize_t n = 0; if (status) *status = SANE_STATUS_GOOD; n = ipc_send (self->fd, self->id, TYPE_CLOSE, 0, NULL); self->id = 0; self->fd = -1; if (0 != n) { if (status) *status = SANE_STATUS_IO_ERROR; log_info ("failed to close network scanner: %s", self->name + strlen ("net:")); return; } log_info ("closed network scanner: %s", self->name + strlen ("net:")); }