/* Copyright (C) 2008, Panasonic Russia Ltd. Copyright (C) 2010, m. allan noah */ /* Panasonic KV-S20xx USB-SCSI scanners. */ #define DEBUG_NOT_STATIC #define BUILD 2 #include "../include/sane/config.h" #include <string.h> #include <unistd.h> #include "../include/sane/sanei_backend.h" #include "../include/sane/sanei_scsi.h" #include "../include/sane/sanei_usb.h" #include "../include/sane/saneopts.h" #include "../include/sane/sanei_config.h" #include "../include/lassert.h" #include "kvs20xx.h" #include "kvs20xx_cmd.h" struct known_device { const SANE_Int id; const SANE_Device scanner; }; static const struct known_device known_devices[] = { { KV_S2025C, { "", "MATSHITA", "KV-S2025C", "sheetfed scanner" }, }, { KV_S2045C, { "", "MATSHITA", "KV-S2045C", "sheetfed scanner" }, }, { KV_S2026C, { "", "MATSHITA", "KV-S2026C", "sheetfed scanner" }, }, { KV_S2046C, { "", "MATSHITA", "KV-S2046C", "sheetfed scanner" }, }, { KV_S2028C, { "", "MATSHITA", "KV-S2028C", "sheetfed scanner" }, }, { KV_S2048C, { "", "MATSHITA", "KV-S2048C", "sheetfed scanner" }, }, }; SANE_Status sane_init (SANE_Int __sane_unused__ * version_code, SANE_Auth_Callback __sane_unused__ authorize) { DBG_INIT (); DBG (DBG_INFO, "This is panasonic kvs20xx driver\n"); *version_code = SANE_VERSION_CODE (V_MAJOR, V_MINOR, BUILD); /* Initialize USB */ sanei_usb_init (); return SANE_STATUS_GOOD; } /* * List of available devices, allocated by sane_get_devices, released * by sane_exit() */ static SANE_Device **devlist = NULL; static unsigned curr_scan_dev = 0; void sane_exit (void) { if (devlist) { int i; for (i = 0; devlist[i]; i++) { free ((void *) devlist[i]->name); free ((void *) devlist[i]); } free ((void *) devlist); devlist = NULL; } } static SANE_Status attach (SANE_String_Const devname) { int i = 0; if (devlist) { for (; devlist[i]; i++); devlist = realloc (devlist, sizeof (SANE_Device *) * (i + 1)); if (!devlist) return SANE_STATUS_NO_MEM; } else { devlist = malloc (sizeof (SANE_Device *) * 2); if (!devlist) return SANE_STATUS_NO_MEM; } devlist[i] = malloc (sizeof (SANE_Device)); if (!devlist[i]) return SANE_STATUS_NO_MEM; memcpy (devlist[i], &known_devices[curr_scan_dev].scanner, sizeof (SANE_Device)); devlist[i]->name = strdup (devname); /* terminate device list with NULL entry: */ devlist[i + 1] = 0; DBG (DBG_INFO, "%s device attached\n", devname); return SANE_STATUS_GOOD; } /* Get device list */ SANE_Status sane_get_devices (const SANE_Device *** device_list, SANE_Bool __sane_unused__ local_only) { if (devlist) { int i; for (i = 0; devlist[i]; i++) { free ((void *) devlist[i]->name); free ((void *) devlist[i]); } free ((void *) devlist); devlist = NULL; } for (curr_scan_dev = 0; curr_scan_dev < sizeof (known_devices) / sizeof (known_devices[0]); curr_scan_dev++) { sanei_usb_find_devices (PANASONIC_ID, known_devices[curr_scan_dev].id, attach); } for (curr_scan_dev = 0; curr_scan_dev < sizeof (known_devices) / sizeof (known_devices[0]); curr_scan_dev++) { sanei_scsi_find_devices (known_devices[curr_scan_dev].scanner.vendor, known_devices[curr_scan_dev].scanner.model, NULL, -1, -1, -1, -1, attach); } if(device_list) *device_list = (const SANE_Device **) devlist; return SANE_STATUS_GOOD; } /* Open device, return the device handle */ SANE_Status sane_open (SANE_String_Const devname, SANE_Handle * handle) { unsigned i, j, id = 0; struct scanner *s; SANE_Int h, bus; SANE_Status st; if (!devlist) { st = sane_get_devices (NULL, 0); if (st) return st; } for (i = 0; devlist[i]; i++) { if (!strcmp (devlist[i]->name, devname)) break; } if (!devlist[i]) return SANE_STATUS_INVAL; for (j = 0; j < sizeof (known_devices) / sizeof (known_devices[0]); j++) { if (!strcmp (devlist[i]->model, known_devices[j].scanner.model)) { id = known_devices[j].id; break; } } st = sanei_usb_open (devname, &h); if (st == SANE_STATUS_ACCESS_DENIED) return st; if (st) { st = sanei_scsi_open (devname, &h, kvs20xx_sense_handler, NULL); if (st) { return st; } bus = SCSI; } else { bus = USB; st = sanei_usb_claim_interface (h, 0); if (st) { sanei_usb_close (h); return st; } } s = malloc (sizeof (struct scanner)); if (!s) return SANE_STATUS_NO_MEM; memset (s, 0, sizeof (struct scanner)); s->buffer = malloc (MAX_READ_DATA_SIZE + BULK_HEADER_SIZE); if (!s->buffer) return SANE_STATUS_NO_MEM; s->file = h; s->bus = bus; s->id = id; kvs20xx_init_options (s); *handle = s; for (i = 0; i < 3; i++) { st = kvs20xx_test_unit_ready (s); if (st) { if (s->bus == SCSI) { sanei_scsi_close (s->file); st = sanei_scsi_open (devname, &h, kvs20xx_sense_handler, NULL); if (st) return st; } else { sanei_usb_release_interface (s->file, 0); sanei_usb_close (s->file); st = sanei_usb_open (devname, &h); if (st) return st; st = sanei_usb_claim_interface (h, 0); if (st) { sanei_usb_close (h); return st; } } s->file = h; } else break; } if (i == 3) return SANE_STATUS_DEVICE_BUSY; st = kvs20xx_set_timeout (s, s->val[FEED_TIMEOUT].w); if (st) { sane_close (s); return st; } return SANE_STATUS_GOOD; } /* Close device */ void sane_close (SANE_Handle handle) { struct scanner *s = (struct scanner *) handle; int i; if (s->bus == USB) { sanei_usb_release_interface (s->file, 0); sanei_usb_close (s->file); } else sanei_scsi_close (s->file); for (i = 1; i < NUM_OPTIONS; i++) { if (s->opt[i].type == SANE_TYPE_STRING && s->val[i].s) free (s->val[i].s); } if (s->data) free (s->data); free (s->buffer); free (s); } /* Get option descriptor */ const SANE_Option_Descriptor * sane_get_option_descriptor (SANE_Handle handle, SANE_Int option) { struct scanner *s = handle; if ((unsigned) option >= NUM_OPTIONS || option < 0) return NULL; return s->opt + option; } static SANE_Status wait_document (struct scanner *s) { SANE_Status st; int i; if (!strcmp ("off", s->val[MANUALFEED].s)) return kvs20xx_document_exist (s); for (i = 0; i < s->val[FEED_TIMEOUT].w; i++) { st = kvs20xx_document_exist (s); if (st != SANE_STATUS_NO_DOCS) return st; sleep (1); } return SANE_STATUS_NO_DOCS; } /* Start scanning */ SANE_Status sane_start (SANE_Handle handle) { struct scanner *s = (struct scanner *) handle; SANE_Status st; int duplex = s->val[DUPLEX].w; if (!s->scanning) { unsigned dummy_length; st = kvs20xx_test_unit_ready (s); if (st) return st; st = wait_document (s); if (st) return st; st = kvs20xx_reset_window (s); if (st) return st; st = kvs20xx_set_window (s, SIDE_FRONT); if (st) return st; if (duplex) { st = kvs20xx_set_window (s, SIDE_BACK); if (st) return st; } st = kvs20xx_scan (s); if (st) return st; st = kvs20xx_read_picture_element (s, SIDE_FRONT, &s->params); if (st) return st; if (duplex) { st = get_adjust_data (s, &dummy_length); if (st) return st; } else { dummy_length = 0; } s->scanning = 1; s->page = 0; s->read = 0; s->side = SIDE_FRONT; sane_get_parameters (s, NULL); s->saved_dummy_size = s->dummy_size = dummy_length ? (dummy_length * s->val[RESOLUTION].w / 1200 - 1) * s->params.bytes_per_line : 0; s->side_size = s->params.lines * s->params.bytes_per_line; s->data = realloc (s->data, duplex ? s->side_size * 2 : s->side_size); if (!s->data) { s->scanning = 0; return SANE_STATUS_NO_MEM; } } if (duplex) { unsigned side = SIDE_FRONT; unsigned read, mx; if (s->side == SIDE_FRONT && s->read == s->side_size - s->dummy_size) { s->side = SIDE_BACK; s->read = s->dummy_size; s->dummy_size = 0; return SANE_STATUS_GOOD; } s->read = 0; s->dummy_size = s->saved_dummy_size; s->side = SIDE_FRONT; st = kvs20xx_document_exist (s); if (st) return st; for (mx = s->side_size * 2; !st; mx -= read, side ^= SIDE_BACK) st = kvs20xx_read_image_data (s, s->page, side, &s->data[s->side_size * 2 - mx], mx, &read); } else { unsigned read, mx; s->read = 0; st = kvs20xx_document_exist (s); if (st) return st; DBG (DBG_INFO, "start: %d\n", s->page); for (mx = s->side_size; !st; mx -= read) st = kvs20xx_read_image_data (s, s->page, SIDE_FRONT, &s->data[s->side_size - mx], mx, &read); } if (st && st != SANE_STATUS_EOF) { s->scanning = 0; return st; } s->page++; return SANE_STATUS_GOOD; } inline static void memcpy24 (u8 * dest, u8 * src, unsigned size, unsigned ls) { unsigned i; for (i = 0; i < size; i++) { dest[i * 3] = src[i]; dest[i * 3 + 1] = src[i + ls]; dest[i * 3 + 2] = src[i + 2 * ls]; } } SANE_Status sane_read (SANE_Handle handle, SANE_Byte * buf, SANE_Int max_len, SANE_Int * len) { struct scanner *s = (struct scanner *) handle; int duplex = s->val[DUPLEX].w; int color = !strcmp (s->val[MODE].s, SANE_VALUE_SCAN_MODE_COLOR); int rest = s->side_size - s->read - s->dummy_size; *len = 0; if (!s->scanning || !rest) { if (strcmp (s->val[FEEDER_MODE].s, SANE_I18N ("continuous"))) { if (!duplex || s->side == SIDE_BACK) s->scanning = 0; } return SANE_STATUS_EOF; } *len = max_len < rest ? max_len : rest; if (duplex && (s->id == KV_S2025C || s->id == KV_S2026C || s->id == KV_S2028C)) { if (color) { unsigned ls = s->params.bytes_per_line; unsigned i, a = s->side == SIDE_FRONT ? 0 : ls / 3; u8 *data; *len = (*len / ls) * ls; for (i = 0, data = s->data + s->read * 2 + a; i < *len / ls; buf += ls, data += 2 * ls, i++) memcpy24 (buf, data, ls / 3, ls * 2 / 3); } else { unsigned ls = s->params.bytes_per_line; unsigned i = s->side == SIDE_FRONT ? 0 : ls; unsigned head = ls - (s->read % ls); unsigned tail = (*len - head) % ls; unsigned lines = (*len - head) / ls; u8 *data = s->data + (s->read / ls) * ls * 2 + i + s->read % ls; assert (data <= s->data + s->side_size * 2); memcpy (buf, data, head); for (i = 0, buf += head, data += head + (head ? ls : 0); i < lines; buf += ls, data += ls * 2, i++) { assert (data <= s->data + s->side_size * 2); memcpy (buf, data, ls); } assert ((data <= s->data + s->side_size * 2) || !tail); memcpy (buf, data, tail); } s->read += *len; } else { if (color) { unsigned i, ls = s->params.bytes_per_line; u8 *data = s->data + s->read; *len = (*len / ls) * ls; for (i = 0; i < *len / ls; buf += ls, data += ls, i++) memcpy24 (buf, data, ls / 3, ls / 3); } else { memcpy (buf, s->data + s->read, *len); } s->read += *len; } return SANE_STATUS_GOOD; } void sane_cancel (SANE_Handle handle) { struct scanner *s = (struct scanner *) handle; s->scanning = 0; } SANE_Status sane_set_io_mode (SANE_Handle __sane_unused__ h, SANE_Bool __sane_unused__ m) { return SANE_STATUS_UNSUPPORTED; } SANE_Status sane_get_select_fd (SANE_Handle __sane_unused__ h, SANE_Int __sane_unused__ * fd) { return SANE_STATUS_UNSUPPORTED; }