/* sane - Scanner Access Now Easy. Copyright (C) 2010-2013 Stéphane Voltz 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, see . */ #define DEBUG_DECLARE_ONLY #include "low.h" #include "assert.h" #include "test_settings.h" #include "gl124_registers.h" #include "gl646_registers.h" #include "gl841_registers.h" #include "gl842_registers.h" #include "gl843_registers.h" #include "gl846_registers.h" #include "gl847_registers.h" #include "gl646_registers.h" #include "gl124.h" #include "gl646.h" #include "gl841.h" #include "gl842.h" #include "gl843.h" #include "gl846.h" #include "gl847.h" #include "gl646.h" #include #include #include #include /* ------------------------------------------------------------------------ */ /* functions calling ASIC specific functions */ /* ------------------------------------------------------------------------ */ namespace genesys { std::unique_ptr create_cmd_set(AsicType asic_type) { switch (asic_type) { case AsicType::GL646: return std::unique_ptr(new gl646::CommandSetGl646{}); case AsicType::GL841: return std::unique_ptr(new gl841::CommandSetGl841{}); case AsicType::GL842: return std::unique_ptr(new gl842::CommandSetGl842{}); case AsicType::GL843: return std::unique_ptr(new gl843::CommandSetGl843{}); case AsicType::GL845: // since only a few reg bits differs we handle both together case AsicType::GL846: return std::unique_ptr(new gl846::CommandSetGl846{}); case AsicType::GL847: return std::unique_ptr(new gl847::CommandSetGl847{}); case AsicType::GL124: return std::unique_ptr(new gl124::CommandSetGl124{}); default: throw SaneException(SANE_STATUS_INVAL, "unknown ASIC type"); } } /* ------------------------------------------------------------------------ */ /* General IO and debugging functions */ /* ------------------------------------------------------------------------ */ void sanei_genesys_write_file(const char* filename, const std::uint8_t* data, std::size_t length) { DBG_HELPER(dbg); std::FILE* out = std::fopen(filename, "w"); if (!out) { throw SaneException("could not open %s for writing: %s", filename, strerror(errno)); } std::fwrite(data, 1, length, out); std::fclose(out); } /* ------------------------------------------------------------------------ */ /* Read and write RAM, registers and AFE */ /* ------------------------------------------------------------------------ */ unsigned sanei_genesys_get_bulk_max_size(AsicType asic_type) { /* Genesys supports 0xFE00 maximum size in general, wheraus GL646 supports 0xFFC0. We use 0xF000 because that's the packet limit in the Linux usbmon USB capture stack. By default it limits packet size to b_size / 5 where b_size is the size of the ring buffer. By default it's 300*1024, so the packet is limited 61440 without any visibility to acquiring software. */ if (asic_type == AsicType::GL124 || asic_type == AsicType::GL846 || asic_type == AsicType::GL847) { return 0xeff0; } return 0xf000; } // Set address for writing data void sanei_genesys_set_buffer_address(Genesys_Device* dev, std::uint32_t addr) { DBG_HELPER(dbg); if (dev->model->asic_type==AsicType::GL847 || dev->model->asic_type==AsicType::GL845 || dev->model->asic_type==AsicType::GL846 || dev->model->asic_type==AsicType::GL124) { DBG(DBG_warn, "%s: shouldn't be used for GL846+ ASICs\n", __func__); return; } DBG(DBG_io, "%s: setting address to 0x%05x\n", __func__, addr & 0xfffffff0); addr = addr >> 4; dev->interface->write_register(0x2b, (addr & 0xff)); addr = addr >> 8; dev->interface->write_register(0x2a, (addr & 0xff)); } /* ------------------------------------------------------------------------ */ /* Medium level functions */ /* ------------------------------------------------------------------------ */ Status scanner_read_status(Genesys_Device& dev) { DBG_HELPER(dbg); std::uint16_t address = 0; switch (dev.model->asic_type) { case AsicType::GL124: address = 0x101; break; case AsicType::GL646: case AsicType::GL841: case AsicType::GL842: case AsicType::GL843: case AsicType::GL845: case AsicType::GL846: case AsicType::GL847: address = 0x41; break; default: throw SaneException("Unsupported asic type"); } // same for all chips constexpr std::uint8_t PWRBIT = 0x80; constexpr std::uint8_t BUFEMPTY = 0x40; constexpr std::uint8_t FEEDFSH = 0x20; constexpr std::uint8_t SCANFSH = 0x10; constexpr std::uint8_t HOMESNR = 0x08; constexpr std::uint8_t LAMPSTS = 0x04; constexpr std::uint8_t FEBUSY = 0x02; constexpr std::uint8_t MOTORENB = 0x01; auto value = dev.interface->read_register(address); Status status; status.is_replugged = !(value & PWRBIT); status.is_buffer_empty = value & BUFEMPTY; status.is_feeding_finished = value & FEEDFSH; status.is_scanning_finished = value & SCANFSH; status.is_at_home = value & HOMESNR; status.is_lamp_on = value & LAMPSTS; status.is_front_end_busy = value & FEBUSY; status.is_motor_enabled = value & MOTORENB; if (DBG_LEVEL >= DBG_io) { debug_print_status(dbg, status); } return status; } Status scanner_read_reliable_status(Genesys_Device& dev) { DBG_HELPER(dbg); scanner_read_status(dev); dev.interface->sleep_ms(100); return scanner_read_status(dev); } void scanner_read_print_status(Genesys_Device& dev) { scanner_read_status(dev); } /** * decodes and prints content of status register * @param val value read from status register */ void debug_print_status(DebugMessageHelper& dbg, Status val) { std::stringstream str; str << val; dbg.vlog(DBG_info, "status=%s\n", str.str().c_str()); } void scanner_register_rw_clear_bits(Genesys_Device& dev, std::uint16_t address, std::uint8_t mask) { scanner_register_rw_bits(dev, address, 0x00, mask); } void scanner_register_rw_set_bits(Genesys_Device& dev, std::uint16_t address, std::uint8_t mask) { scanner_register_rw_bits(dev, address, mask, mask); } void scanner_register_rw_bits(Genesys_Device& dev, std::uint16_t address, std::uint8_t value, std::uint8_t mask) { auto reg_value = dev.interface->read_register(address); reg_value = (reg_value & ~mask) | (value & mask); dev.interface->write_register(address, reg_value); } /** read the number of valid words in scanner's RAM * ie registers 42-43-44 */ // candidate for moving into chip specific files? void sanei_genesys_read_valid_words(Genesys_Device* dev, unsigned int* words) { DBG_HELPER(dbg); switch (dev->model->asic_type) { case AsicType::GL124: *words = dev->interface->read_register(0x102) & 0x03; *words = *words * 256 + dev->interface->read_register(0x103); *words = *words * 256 + dev->interface->read_register(0x104); *words = *words * 256 + dev->interface->read_register(0x105); break; case AsicType::GL845: case AsicType::GL846: *words = dev->interface->read_register(0x42) & 0x02; *words = *words * 256 + dev->interface->read_register(0x43); *words = *words * 256 + dev->interface->read_register(0x44); *words = *words * 256 + dev->interface->read_register(0x45); break; case AsicType::GL847: *words = dev->interface->read_register(0x42) & 0x03; *words = *words * 256 + dev->interface->read_register(0x43); *words = *words * 256 + dev->interface->read_register(0x44); *words = *words * 256 + dev->interface->read_register(0x45); break; default: *words = dev->interface->read_register(0x44); *words += dev->interface->read_register(0x43) * 256; if (dev->model->asic_type == AsicType::GL646) { *words += ((dev->interface->read_register(0x42) & 0x03) * 256 * 256); } else { *words += ((dev->interface->read_register(0x42) & 0x0f) * 256 * 256); } } DBG(DBG_proc, "%s: %d words\n", __func__, *words); } /** read the number of lines scanned * ie registers 4b-4c-4d */ void sanei_genesys_read_scancnt(Genesys_Device* dev, unsigned int* words) { DBG_HELPER(dbg); if (dev->model->asic_type == AsicType::GL124) { *words = (dev->interface->read_register(0x10b) & 0x0f) << 16; *words += (dev->interface->read_register(0x10c) << 8); *words += dev->interface->read_register(0x10d); } else { *words = dev->interface->read_register(0x4d); *words += dev->interface->read_register(0x4c) * 256; if (dev->model->asic_type == AsicType::GL646) { *words += ((dev->interface->read_register(0x4b) & 0x03) * 256 * 256); } else { *words += ((dev->interface->read_register(0x4b) & 0x0f) * 256 * 256); } } DBG(DBG_proc, "%s: %d lines\n", __func__, *words); } /** @brief Check if the scanner's internal data buffer is empty * @param *dev device to test for data * @param *empty return value * @return empty will be set to true if there is no scanned data. **/ bool sanei_genesys_is_buffer_empty(Genesys_Device* dev) { DBG_HELPER(dbg); dev->interface->sleep_ms(1); auto status = scanner_read_status(*dev); if (status.is_buffer_empty) { /* fix timing issue on USB3 (or just may be too fast) hardware * spotted by John S. Weber */ dev->interface->sleep_ms(1); DBG(DBG_io2, "%s: buffer is empty\n", __func__); return true; } DBG(DBG_io, "%s: buffer is filled\n", __func__); return false; } void wait_until_buffer_non_empty(Genesys_Device* dev, bool check_status_twice) { // FIXME: reduce MAX_RETRIES once tests are updated const unsigned MAX_RETRIES = 100000; for (unsigned i = 0; i < MAX_RETRIES; ++i) { if (check_status_twice) { // FIXME: this only to preserve previous behavior, can be removed scanner_read_status(*dev); } bool empty = sanei_genesys_is_buffer_empty(dev); dev->interface->sleep_ms(10); if (!empty) return; } throw SaneException(SANE_STATUS_IO_ERROR, "failed to read data"); } void wait_until_has_valid_words(Genesys_Device* dev) { unsigned words = 0; unsigned sleep_time_ms = 10; for (unsigned wait_ms = 0; wait_ms < 70000; wait_ms += sleep_time_ms) { sanei_genesys_read_valid_words(dev, &words); if (words != 0) break; dev->interface->sleep_ms(sleep_time_ms); } if (words == 0) { throw SaneException(SANE_STATUS_IO_ERROR, "timeout, buffer does not get filled"); } } // Read data (e.g scanned image) from scan buffer void sanei_genesys_read_data_from_scanner(Genesys_Device* dev, std::uint8_t* data, size_t size) { DBG_HELPER_ARGS(dbg, "size = %zu bytes", size); if (size & 1) DBG(DBG_info, "WARNING %s: odd number of bytes\n", __func__); wait_until_has_valid_words(dev); dev->interface->bulk_read_data(0x45, data, size); } Image read_unshuffled_image_from_scanner(Genesys_Device* dev, const ScanSession& session, std::size_t total_bytes) { DBG_HELPER(dbg); auto format = create_pixel_format(session.params.depth, dev->model->is_cis ? 1 : session.params.channels, dev->model->line_mode_color_order); auto width = get_pixels_from_row_bytes(format, session.output_line_bytes_raw); auto height = session.optical_line_count; Image image(width, height, format); auto max_bytes = image.get_row_bytes() * height; if (total_bytes > max_bytes) { throw SaneException("Trying to read too much data %zu (max %zu)", total_bytes, max_bytes); } if (total_bytes != max_bytes) { DBG(DBG_info, "WARNING %s: trying to read not enough data (%zu, full fill %zu)\n", __func__, total_bytes, max_bytes); } sanei_genesys_read_data_from_scanner(dev, image.get_row_ptr(0), total_bytes); ImagePipelineStack pipeline; pipeline.push_first_node(image); if (session.segment_count > 1) { auto output_width = session.output_segment_pixel_group_count * session.segment_count; pipeline.push_node(output_width, dev->segment_order, session.conseq_pixel_dist, 1, 1); } if (session.params.depth == 16) { unsigned num_swaps = 0; if (has_flag(dev->model->flags, ModelFlag::SWAP_16BIT_DATA)) { num_swaps++; } #ifdef WORDS_BIGENDIAN num_swaps++; #endif if (num_swaps % 2 != 0) { dev->pipeline.push_node(); } } if (has_flag(dev->model->flags, ModelFlag::INVERT_PIXEL_DATA)) { pipeline.push_node(); } if (dev->model->is_cis && session.params.channels == 3) { pipeline.push_node(dev->model->line_mode_color_order); } if (session.use_host_side_gray) { pipeline.push_node(); } if (pipeline.get_output_format() == PixelFormat::BGR888) { pipeline.push_node(PixelFormat::RGB888); } if (pipeline.get_output_format() == PixelFormat::BGR161616) { pipeline.push_node(PixelFormat::RGB161616); } return pipeline.get_image(); } Image read_shuffled_image_from_scanner(Genesys_Device* dev, const ScanSession& session) { DBG_HELPER(dbg); std::size_t total_bytes = 0; std::size_t pixels_per_line = 0; if (dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843 || dev->model->model_id == ModelId::CANON_5600F) { pixels_per_line = session.output_pixels; } else { // BUG: this selects incorrect pixel number pixels_per_line = session.params.pixels; } // FIXME: the current calculation is likely incorrect on non-GL843 implementations, // but this needs checking. Note the extra line when computing size. if (dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843 || dev->model->model_id == ModelId::CANON_5600F) { total_bytes = session.output_total_bytes_raw; } else { total_bytes = session.params.channels * 2 * pixels_per_line * (session.params.lines + 1); } auto format = create_pixel_format(session.params.depth, dev->model->is_cis ? 1 : session.params.channels, dev->model->line_mode_color_order); // auto width = get_pixels_from_row_bytes(format, session.output_line_bytes_raw); auto width = pixels_per_line; auto height = session.params.lines + 1; // BUG: incorrect if (dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843 || dev->model->model_id == ModelId::CANON_5600F) { height = session.optical_line_count; } Image image(width, height, format); auto max_bytes = image.get_row_bytes() * height; if (total_bytes > max_bytes) { throw SaneException("Trying to read too much data %zu (max %zu)", total_bytes, max_bytes); } if (total_bytes != max_bytes) { DBG(DBG_info, "WARNING %s: trying to read not enough data (%zu, full fill %zu)\n", __func__, total_bytes, max_bytes); } sanei_genesys_read_data_from_scanner(dev, image.get_row_ptr(0), total_bytes); ImagePipelineStack pipeline; pipeline.push_first_node(image); if (session.segment_count > 1) { auto output_width = session.output_segment_pixel_group_count * session.segment_count; pipeline.push_node(output_width, dev->segment_order, session.conseq_pixel_dist, 1, 1); } if (session.params.depth == 16) { unsigned num_swaps = 0; if (has_flag(dev->model->flags, ModelFlag::SWAP_16BIT_DATA)) { num_swaps++; } #ifdef WORDS_BIGENDIAN num_swaps++; #endif if (num_swaps % 2 != 0) { dev->pipeline.push_node(); } } if (has_flag(dev->model->flags, ModelFlag::INVERT_PIXEL_DATA)) { pipeline.push_node(); } if (dev->model->is_cis && session.params.channels == 3) { pipeline.push_node(dev->model->line_mode_color_order); } if (pipeline.get_output_format() == PixelFormat::BGR888) { pipeline.push_node(PixelFormat::RGB888); } if (pipeline.get_output_format() == PixelFormat::BGR161616) { pipeline.push_node(PixelFormat::RGB161616); } return pipeline.get_image(); } void sanei_genesys_read_feed_steps(Genesys_Device* dev, unsigned int* steps) { DBG_HELPER(dbg); if (dev->model->asic_type == AsicType::GL124) { *steps = (dev->interface->read_register(0x108) & 0x1f) << 16; *steps += (dev->interface->read_register(0x109) << 8); *steps += dev->interface->read_register(0x10a); } else { *steps = dev->interface->read_register(0x4a); *steps += dev->interface->read_register(0x49) * 256; if (dev->model->asic_type == AsicType::GL646) { *steps += ((dev->interface->read_register(0x48) & 0x03) * 256 * 256); } else if (dev->model->asic_type == AsicType::GL841) { *steps += ((dev->interface->read_register(0x48) & 0x0f) * 256 * 256); } else { *steps += ((dev->interface->read_register(0x48) & 0x1f) * 256 * 256); } } DBG(DBG_proc, "%s: %d steps\n", __func__, *steps); } void sanei_genesys_set_lamp_power(Genesys_Device* dev, const Genesys_Sensor& sensor, Genesys_Register_Set& regs, bool set) { static const std::uint8_t REG_0x03_LAMPPWR = 0x10; if (set) { regs.find_reg(0x03).value |= REG_0x03_LAMPPWR; if (dev->model->asic_type == AsicType::GL841) { regs_set_exposure(dev->model->asic_type, regs, sanei_genesys_fixup_exposure(sensor.exposure)); regs.set8(0x19, 0x50); } if (dev->model->asic_type == AsicType::GL843) { regs_set_exposure(dev->model->asic_type, regs, sensor.exposure); } // we don't actually turn on lamp on infrared scan if ((dev->model->model_id == ModelId::CANON_8400F || dev->model->model_id == ModelId::CANON_8600F || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7200I || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7500I || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_8200I) && dev->settings.scan_method == ScanMethod::TRANSPARENCY_INFRARED) { regs.find_reg(0x03).value &= ~REG_0x03_LAMPPWR; } } else { regs.find_reg(0x03).value &= ~REG_0x03_LAMPPWR; if (dev->model->asic_type == AsicType::GL841) { regs_set_exposure(dev->model->asic_type, regs, sanei_genesys_fixup_exposure({0, 0, 0})); regs.set8(0x19, 0xff); } if (dev->model->model_id == ModelId::CANON_5600F) { regs_set_exposure(dev->model->asic_type, regs, sanei_genesys_fixup_exposure({0, 0, 0})); } } regs.state.is_lamp_on = set; } void sanei_genesys_set_motor_power(Genesys_Register_Set& regs, bool set) { static const std::uint8_t REG_0x02_MTRPWR = 0x10; if (set) { regs.find_reg(0x02).value |= REG_0x02_MTRPWR; } else { regs.find_reg(0x02).value &= ~REG_0x02_MTRPWR; } regs.state.is_motor_on = set; } bool should_enable_gamma(const ScanSession& session, const Genesys_Sensor& sensor) { if ((session.params.flags & ScanFlag::DISABLE_GAMMA) != ScanFlag::NONE) { return false; } if (session.params.depth == 16) { return false; } if (session.params.brightness_adjustment != 0 || session.params.contrast_adjustment != 0) { return true; } if (sensor.gamma[0] == 1.0f || sensor.gamma[1] == 1.0f || sensor.gamma[2] == 1.0f) { return false; } return true; } std::vector get_gamma_table(Genesys_Device* dev, const Genesys_Sensor& sensor, int color) { if (!dev->gamma_override_tables[color].empty()) { return dev->gamma_override_tables[color]; } else { std::vector ret; sanei_genesys_create_default_gamma_table(dev, ret, sensor.gamma[color]); return ret; } } /** @brief generates gamma buffer to transfer * Generates gamma table buffer to send to ASIC. Applies * contrast and brightness if set. * @param dev device to set up * @param bits number of bits used by gamma * @param max value for gamma * @param size of the gamma table */ std::vector generate_gamma_buffer(Genesys_Device* dev, const Genesys_Sensor& sensor, int bits, int max, int size) { DBG_HELPER(dbg); // the gamma tables are 16 bits words and contain 3 channels std::vector gamma_buf(size * 2 * 3); std::vector rgamma = get_gamma_table(dev, sensor, GENESYS_RED); std::vector ggamma = get_gamma_table(dev, sensor, GENESYS_GREEN); std::vector bgamma = get_gamma_table(dev, sensor, GENESYS_BLUE); auto get_gamma_value = [](const std::vector& array, std::size_t index) -> std::uint16_t { if (index < array.size()) return array[index]; return 0xffff; }; auto set_gamma_buf_value = [](std::vector& array, std::size_t pos, std::uint16_t value) { array[pos * 2 + 0] = value & 0xff; array[pos * 2 + 1] = (value >> 8) & 0xff; }; if(dev->settings.contrast!=0 || dev->settings.brightness!=0) { std::vector lut(65536); sanei_genesys_load_lut(reinterpret_cast(lut.data()), bits, bits, 0, max, dev->settings.contrast, dev->settings.brightness); for (int i = 0; i < size; i++) { set_gamma_buf_value(gamma_buf, i + size * 0, lut[get_gamma_value(rgamma, i)]); set_gamma_buf_value(gamma_buf, i + size * 1, lut[get_gamma_value(ggamma, i)]); set_gamma_buf_value(gamma_buf, i + size * 2, lut[get_gamma_value(bgamma, i)]); } } else { for (int i = 0; i < size; i++) { set_gamma_buf_value(gamma_buf, i + size * 0, get_gamma_value(rgamma, i)); set_gamma_buf_value(gamma_buf, i + size * 1, get_gamma_value(ggamma, i)); set_gamma_buf_value(gamma_buf, i + size * 2, get_gamma_value(bgamma, i)); } } return gamma_buf; } /** @brief send gamma table to scanner * This function sends generic gamma table (ie ones built with * provided gamma) or the user defined one if provided by * fontend. Used by gl846+ ASICs * @param dev device to write to */ void sanei_genesys_send_gamma_table(Genesys_Device* dev, const Genesys_Sensor& sensor) { DBG_HELPER(dbg); int size; int i; size = 256 + 1; auto gamma = generate_gamma_buffer(dev, sensor, 16, 65535, size); // loop sending gamma tables NOTE: 0x01000000 not 0x10000000 for (i = 0; i < 3; i++) { // clear corresponding GMM_N bit std::uint8_t val = dev->interface->read_register(0xbd); val &= ~(0x01 << i); dev->interface->write_register(0xbd, val); // clear corresponding GMM_F bit val = dev->interface->read_register(0xbe); val &= ~(0x01 << i); dev->interface->write_register(0xbe, val); // FIXME: currently the last word of each gamma table is not initialized, so to work around // unstable data, just set it to 0 which is the most likely value of uninitialized memory // (proper value is probably 0xff) gamma[size * 2 * i + size * 2 - 2] = 0; gamma[size * 2 * i + size * 2 - 1] = 0; /* set GMM_Z */ dev->interface->write_register(0xc5+2*i, gamma[size*2*i+1]); dev->interface->write_register(0xc6+2*i, gamma[size*2*i]); dev->interface->write_ahb(0x01000000 + 0x200 * i, (size-1) * 2, gamma.data() + i * size * 2+2); } } void compute_session_pixel_offsets(const Genesys_Device* dev, ScanSession& s, const Genesys_Sensor& sensor) { if (dev->model->asic_type == AsicType::GL646) { s.pixel_startx += s.output_startx * sensor.full_resolution / s.params.xres; s.pixel_endx = s.pixel_startx + s.optical_pixels * s.full_resolution / s.optical_resolution; } else if (dev->model->asic_type == AsicType::GL841 || dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843 || dev->model->asic_type == AsicType::GL845 || dev->model->asic_type == AsicType::GL846 || dev->model->asic_type == AsicType::GL847) { unsigned startx_xres = s.optical_resolution; if (dev->model->model_id == ModelId::CANON_5600F || dev->model->model_id == ModelId::CANON_LIDE_90) { if (s.output_resolution == 1200) { startx_xres /= 2; } if (s.output_resolution >= 2400) { startx_xres /= 4; } } s.pixel_startx = (s.output_startx * startx_xres) / s.params.xres; s.pixel_endx = s.pixel_startx + s.optical_pixels_raw; } else if (dev->model->asic_type == AsicType::GL124) { s.pixel_startx = s.output_startx * sensor.full_resolution / s.params.xres; s.pixel_endx = s.pixel_startx + s.optical_pixels_raw; } // align pixels to correct boundary for unstaggering unsigned needed_x_alignment = std::max(s.stagger_x.size(), s.stagger_y.size()); unsigned aligned_pixel_startx = align_multiple_floor(s.pixel_startx, needed_x_alignment); s.pixel_endx -= s.pixel_startx - aligned_pixel_startx; s.pixel_startx = aligned_pixel_startx; s.pixel_startx = sensor.pixel_count_ratio.apply(s.pixel_startx); s.pixel_endx = sensor.pixel_count_ratio.apply(s.pixel_endx); if (dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7200 || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7200I || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7300 || dev->model->model_id == ModelId::PLUSTEK_OPTICFILM_7500I) { s.pixel_startx = align_multiple_floor(s.pixel_startx, sensor.pixel_count_ratio.divisor()); s.pixel_endx = align_multiple_floor(s.pixel_endx, sensor.pixel_count_ratio.divisor()); } } unsigned session_adjust_output_pixels(unsigned output_pixels, const Genesys_Device& dev, const Genesys_Sensor& sensor, unsigned output_xresolution, unsigned output_yresolution, bool adjust_output_pixels) { bool adjust_optical_pixels = !adjust_output_pixels; if (dev.model->model_id == ModelId::CANON_5600F) { adjust_optical_pixels = true; adjust_output_pixels = true; } if (adjust_optical_pixels) { auto optical_resolution = sensor.get_optical_resolution(); // FIXME: better way would be to compute and return the required multiplier unsigned optical_pixels = (output_pixels * optical_resolution) / output_xresolution; if (dev.model->asic_type == AsicType::GL841 || dev.model->asic_type == AsicType::GL842) { optical_pixels = align_multiple_ceil(optical_pixels, 2); } if (dev.model->asic_type == AsicType::GL646 && output_xresolution == 400) { optical_pixels = align_multiple_floor(optical_pixels, 6); } if (dev.model->asic_type == AsicType::GL843) { // ensure the number of optical pixels is divisible by 2. // In quarter-CCD mode optical_pixels is 4x larger than the actual physical number optical_pixels = align_multiple_ceil(optical_pixels, 2 * sensor.full_resolution / optical_resolution); if (dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_7200 || dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_7200I || dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_7300 || dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_7400 || dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_7500I || dev.model->model_id == ModelId::PLUSTEK_OPTICFILM_8200I) { optical_pixels = align_multiple_ceil(optical_pixels, 16); } } output_pixels = (optical_pixels * output_xresolution) / optical_resolution; } if (adjust_output_pixels) { // TODO: the following may no longer be needed but were applied historically. // we need an even pixels number // TODO invert test logic or generalize behaviour across all ASICs if (has_flag(dev.model->flags, ModelFlag::SIS_SENSOR) || dev.model->asic_type == AsicType::GL847 || dev.model->asic_type == AsicType::GL124 || dev.model->asic_type == AsicType::GL845 || dev.model->asic_type == AsicType::GL846 || dev.model->asic_type == AsicType::GL843) { if (output_xresolution <= 1200) { output_pixels = align_multiple_floor(output_pixels, 4); } else if (output_xresolution < output_yresolution) { // BUG: this is an artifact of the fact that the resolution was twice as large than // the actual resolution when scanning above the supported scanner X resolution output_pixels = align_multiple_floor(output_pixels, 8); } else { output_pixels = align_multiple_floor(output_pixels, 16); } } // corner case for true lineart for sensor with several segments or when xres is doubled // to match yres */ if (output_xresolution >= 1200 && ( dev.model->asic_type == AsicType::GL124 || dev.model->asic_type == AsicType::GL847 || dev.session.params.xres < dev.session.params.yres)) { if (output_xresolution < output_yresolution) { // FIXME: this is an artifact of the fact that the resolution was twice as large than // the actual resolution when scanning above the supported scanner X resolution output_pixels = align_multiple_floor(output_pixels, 8); } else { output_pixels = align_multiple_floor(output_pixels, 16); } } } return output_pixels; } void compute_session(const Genesys_Device* dev, ScanSession& s, const Genesys_Sensor& sensor) { DBG_HELPER(dbg); (void) dev; s.params.assert_valid(); if (s.params.depth != 8 && s.params.depth != 16) { throw SaneException("Unsupported depth setting %d", s.params.depth); } // compute optical and output resolutions s.full_resolution = sensor.full_resolution; s.optical_resolution = sensor.get_optical_resolution(); s.output_resolution = s.params.xres; s.pixel_count_ratio = sensor.pixel_count_ratio; if (s.output_resolution > s.optical_resolution) { throw std::runtime_error("output resolution higher than optical resolution"); } s.output_pixels = session_adjust_output_pixels(s.params.pixels, *dev, sensor, s.params.xres, s.params.yres, false); // Compute the number of optical pixels that will be acquired by the chip. // The necessary alignment requirements have already been computed by // get_session_output_pixels_multiplier s.optical_pixels = (s.output_pixels * s.optical_resolution) / s.output_resolution; if (static_cast(s.params.startx) + sensor.output_pixel_offset < 0) throw SaneException("Invalid sensor.output_pixel_offset"); s.output_startx = static_cast( static_cast(s.params.startx) + sensor.output_pixel_offset); if (has_flag(dev->model->flags, ModelFlag::HOST_SIDE_GRAY) && s.params.channels == 1 && s.params.color_filter == ColorFilter::NONE) { s.use_host_side_gray = true; s.params.channels = 3; s.params.scan_mode = ScanColorMode::COLOR_SINGLE_PASS; } s.stagger_x = sensor.stagger_x; s.stagger_y = sensor.stagger_y; s.num_staggered_lines = 0; if (!has_flag(s.params.flags, ScanFlag::IGNORE_STAGGER_OFFSET)) { s.num_staggered_lines = s.stagger_y.max_shift() * s.params.yres / s.params.xres; } s.color_shift_lines_r = dev->model->ld_shift_r; s.color_shift_lines_g = dev->model->ld_shift_g; s.color_shift_lines_b = dev->model->ld_shift_b; if (dev->model->motor_id == MotorId::G4050 && s.params.yres > 600) { // it seems base_dpi of the G4050 motor is changed above 600 dpi s.color_shift_lines_r = (s.color_shift_lines_r * 3800) / dev->motor.base_ydpi; s.color_shift_lines_g = (s.color_shift_lines_g * 3800) / dev->motor.base_ydpi; s.color_shift_lines_b = (s.color_shift_lines_b * 3800) / dev->motor.base_ydpi; } s.color_shift_lines_r = (s.color_shift_lines_r * s.params.yres) / dev->motor.base_ydpi; s.color_shift_lines_g = (s.color_shift_lines_g * s.params.yres) / dev->motor.base_ydpi; s.color_shift_lines_b = (s.color_shift_lines_b * s.params.yres) / dev->motor.base_ydpi; s.max_color_shift_lines = 0; if (s.params.channels > 1 && !has_flag(s.params.flags, ScanFlag::IGNORE_COLOR_OFFSET)) { s.max_color_shift_lines = std::max(s.color_shift_lines_r, std::max(s.color_shift_lines_g, s.color_shift_lines_b)); } s.output_line_count = s.params.lines + s.max_color_shift_lines + s.num_staggered_lines; s.optical_line_count = dev->model->is_cis ? s.output_line_count * s.params.channels : s.output_line_count; s.output_channel_bytes = multiply_by_depth_ceil(s.output_pixels, s.params.depth); s.output_line_bytes = s.output_channel_bytes * s.params.channels; s.segment_count = sensor.get_segment_count(); s.optical_pixels_raw = s.optical_pixels; s.output_line_bytes_raw = s.output_line_bytes; s.conseq_pixel_dist = 0; // FIXME: Use ModelFlag::SIS_SENSOR if ((dev->model->asic_type == AsicType::GL845 || dev->model->asic_type == AsicType::GL846 || dev->model->asic_type == AsicType::GL847) && dev->model->model_id != ModelId::PLUSTEK_OPTICFILM_7400 && dev->model->model_id != ModelId::PLUSTEK_OPTICFILM_8200I) { if (s.segment_count > 1) { s.conseq_pixel_dist = sensor.segment_size; // in case of multi-segments sensor, we have expand the scan area to sensor boundary if (dev->model->model_id == ModelId::CANON_5600F) { unsigned startx_xres = s.optical_resolution; if (dev->model->model_id == ModelId::CANON_5600F) { if (s.output_resolution == 1200) { startx_xres /= 2; } if (s.output_resolution >= 2400) { startx_xres /= 4; } } unsigned optical_startx = s.output_startx * startx_xres / s.params.xres; unsigned optical_endx = optical_startx + s.optical_pixels; unsigned multi_segment_size_output = s.segment_count * s.conseq_pixel_dist; unsigned multi_segment_size_optical = (multi_segment_size_output * s.optical_resolution) / s.output_resolution; optical_endx = align_multiple_ceil(optical_endx, multi_segment_size_optical); s.optical_pixels_raw = optical_endx - optical_startx; s.optical_pixels_raw = align_multiple_floor(s.optical_pixels_raw, 4 * s.optical_resolution / s.output_resolution); } else { // BUG: the following code will likely scan too much. Use the CANON_5600F approach unsigned extra_segment_scan_area = align_multiple_ceil(s.conseq_pixel_dist, 2); extra_segment_scan_area *= s.segment_count - 1; extra_segment_scan_area = s.pixel_count_ratio.apply_inverse(extra_segment_scan_area); s.optical_pixels_raw += extra_segment_scan_area; } } if (dev->model->model_id == ModelId::CANON_5600F) { auto output_pixels_raw = (s.optical_pixels_raw * s.output_resolution) / s.optical_resolution; auto output_channel_bytes_raw = multiply_by_depth_ceil(output_pixels_raw, s.params.depth); s.output_line_bytes_raw = output_channel_bytes_raw * s.params.channels; } else { s.output_line_bytes_raw = multiply_by_depth_ceil( (s.optical_pixels_raw * s.output_resolution) / sensor.full_resolution / s.segment_count, s.params.depth); } } if (dev->model->asic_type == AsicType::GL841 || dev->model->asic_type == AsicType::GL842) { if (dev->model->is_cis) { s.output_line_bytes_raw = s.output_channel_bytes; } } if (dev->model->asic_type == AsicType::GL124) { if (dev->model->is_cis) { s.output_line_bytes_raw = s.output_channel_bytes; } s.conseq_pixel_dist = s.output_pixels / (s.full_resolution / s.optical_resolution) / s.segment_count; } if (dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843) { if (dev->model->is_cis) { if (s.segment_count > 1) { s.conseq_pixel_dist = sensor.segment_size; } } else { s.conseq_pixel_dist = s.output_pixels / s.segment_count; } } s.output_segment_pixel_group_count = 0; if (dev->model->asic_type == AsicType::GL124 || dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843) { s.output_segment_pixel_group_count = s.output_pixels / (s.full_resolution / s.optical_resolution * s.segment_count); } if (dev->model->model_id == ModelId::CANON_LIDE_90) { s.output_segment_pixel_group_count = s.output_pixels / s.segment_count; } if (dev->model->asic_type == AsicType::GL845 || dev->model->asic_type == AsicType::GL846 || dev->model->asic_type == AsicType::GL847) { if (dev->model->model_id == ModelId::CANON_5600F) { s.output_segment_pixel_group_count = s.output_pixels / s.segment_count; } else { s.output_segment_pixel_group_count = s.pixel_count_ratio.apply(s.optical_pixels); } } s.output_line_bytes_requested = multiply_by_depth_ceil( s.params.get_requested_pixels() * s.params.channels, s.params.depth); s.output_total_bytes_raw = s.output_line_bytes_raw * s.output_line_count; s.output_total_bytes = s.output_line_bytes * s.output_line_count; if (dev->model->model_id == ModelId::CANON_LIDE_90) { s.output_total_bytes_raw *= s.params.channels; s.output_total_bytes *= s.params.channels; } s.buffer_size_read = s.output_line_bytes_raw * 64; compute_session_pixel_offsets(dev, s, sensor); s.shading_pixel_offset = sensor.shading_pixel_offset; if (dev->model->asic_type == AsicType::GL124 || dev->model->asic_type == AsicType::GL845 || dev->model->asic_type == AsicType::GL846) { s.enable_ledadd = (s.params.channels == 1 && dev->model->is_cis && s.params.color_filter == ColorFilter::NONE); } s.use_host_side_calib = sensor.use_host_side_calib; if (dev->model->asic_type == AsicType::GL841 || dev->model->asic_type == AsicType::GL842 || dev->model->asic_type == AsicType::GL843) { // no 16 bit gamma for this ASIC if (s.params.depth == 16) { s.params.flags |= ScanFlag::DISABLE_GAMMA; } } s.computed = true; DBG(DBG_info, "%s ", __func__); debug_dump(DBG_info, s); } ImagePipelineStack build_image_pipeline(const Genesys_Device& dev, const ScanSession& session, unsigned pipeline_index, bool log_image_data) { auto format = create_pixel_format(session.params.depth, dev.model->is_cis ? 1 : session.params.channels, dev.model->line_mode_color_order); auto depth = get_pixel_format_depth(format); auto width = get_pixels_from_row_bytes(format, session.output_line_bytes_raw); auto read_data_from_usb = [&dev](std::size_t size, std::uint8_t* data) { DBG(DBG_info, "read_data_from_usb: reading %zu bytes\n", size); auto begin = std::chrono::high_resolution_clock::now(); dev.interface->bulk_read_data(0x45, data, size); auto end = std::chrono::high_resolution_clock::now(); float us = std::chrono::duration_cast(end - begin).count(); float speed = size / us; // bytes/us == MB/s DBG(DBG_info, "read_data_from_usb: reading %zu bytes finished %f MB/s\n", size, speed); return true; }; auto debug_prefix = "gl_pipeline_" + std::to_string(pipeline_index); ImagePipelineStack pipeline; auto lines = session.optical_line_count; auto buffer_size = session.buffer_size_read; // At least GL841 requires reads to be aligned to 2 bytes and will fail on some devices on // certain circumstances. buffer_size = align_multiple_ceil(buffer_size, 2); auto& src_node = pipeline.push_first_node( width, lines, format, buffer_size, read_data_from_usb); src_node.set_last_read_multiple(2); if (log_image_data) { pipeline.push_node(debug_prefix + "_0_from_usb.tiff"); } if (session.segment_count > 1) { auto output_width = session.output_segment_pixel_group_count * session.segment_count; pipeline.push_node(output_width, dev.segment_order, session.conseq_pixel_dist, 1, 1); if (log_image_data) { pipeline.push_node(debug_prefix + "_1_after_desegment.tiff"); } } if (depth == 16) { unsigned num_swaps = 0; if (has_flag(dev.model->flags, ModelFlag::SWAP_16BIT_DATA)) { num_swaps++; } #ifdef WORDS_BIGENDIAN num_swaps++; #endif if (num_swaps % 2 != 0) { pipeline.push_node(); if (log_image_data) { pipeline.push_node(debug_prefix + "_2_after_swap.tiff"); } } } if (has_flag(dev.model->flags, ModelFlag::INVERT_PIXEL_DATA)) { pipeline.push_node(); if (log_image_data) { pipeline.push_node(debug_prefix + "_3_after_invert.tiff"); } } if (dev.model->is_cis && session.params.channels == 3) { pipeline.push_node(dev.model->line_mode_color_order); if (log_image_data) { pipeline.push_node(debug_prefix + "_4_after_merge_mono.tiff"); } } if (pipeline.get_output_format() == PixelFormat::BGR888) { pipeline.push_node(PixelFormat::RGB888); } if (pipeline.get_output_format() == PixelFormat::BGR161616) { pipeline.push_node(PixelFormat::RGB161616); } if (log_image_data) { pipeline.push_node(debug_prefix + "_5_after_format.tiff"); } if (session.max_color_shift_lines > 0 && session.params.channels == 3) { pipeline.push_node( session.color_shift_lines_r, session.color_shift_lines_g, session.color_shift_lines_b); if (log_image_data) { pipeline.push_node(debug_prefix + "_6_after_color_unshift.tiff"); } } if (!session.stagger_x.empty()) { // FIXME: the image will be scaled to requested pixel count without regard to the reduction // of image size in this step. pipeline.push_node(session.stagger_x.shifts()); if (log_image_data) { pipeline.push_node(debug_prefix + "_7_after_x_unstagger.tiff"); } } if (session.num_staggered_lines > 0) { pipeline.push_node(session.stagger_y.shifts()); if (log_image_data) { pipeline.push_node(debug_prefix + "_8_after_y_unstagger.tiff"); } } if (session.use_host_side_calib && !has_flag(dev.model->flags, ModelFlag::DISABLE_SHADING_CALIBRATION) && !has_flag(session.params.flags, ScanFlag::DISABLE_SHADING)) { unsigned offset_pixels = session.params.startx + dev.calib_session.shading_pixel_offset; unsigned offset_bytes = offset_pixels * dev.calib_session.params.channels; pipeline.push_node(dev.dark_average_data, dev.white_average_data, offset_bytes); if (log_image_data) { pipeline.push_node(debug_prefix + "_9_after_calibrate.tiff"); } } if (session.use_host_side_gray) { pipeline.push_node(); if (log_image_data) { pipeline.push_node(debug_prefix + "_10_after_nogray.tiff"); } } if (pipeline.get_output_width() != session.params.get_requested_pixels()) { pipeline.push_node(session.params.get_requested_pixels()); } return pipeline; } void setup_image_pipeline(Genesys_Device& dev, const ScanSession& session) { static unsigned s_pipeline_index = 0; s_pipeline_index++; dev.pipeline = build_image_pipeline(dev, session, s_pipeline_index, dbg_log_image_data()); auto read_from_pipeline = [&dev](std::size_t size, std::uint8_t* out_data) { (void) size; // will be always equal to dev.pipeline.get_output_row_bytes() return dev.pipeline.get_next_row_data(out_data); }; dev.pipeline_buffer = ImageBuffer{dev.pipeline.get_output_row_bytes(), read_from_pipeline}; } std::uint8_t compute_frontend_gain_wolfson(float value, float target_value) { /* the flow of data through the frontend ADC is as follows (see e.g. WM8192 datasheet) input -> apply offset (o = i + 260mV * (DAC[7:0]-127.5)/127.5) -> -> apply gain (o = i * 208/(283-PGA[7:0]) -> ADC Here we have some input data that was acquired with zero gain (PGA==0). We want to compute gain such that the output would approach full ADC range (controlled by target_value). We want to solve the following for {PGA}: {value} = {input} * 208 / (283 - 0) {target_value} = {input} * 208 / (283 - {PGA}) The solution is the following equation: {PGA} = 283 * (1 - {value} / {target_value}) */ float gain = value / target_value; int code = static_cast(283 * (1 - gain)); return clamp(code, 0, 255); } std::uint8_t compute_frontend_gain_lide_80(float value, float target_value) { int code = static_cast((target_value / value) * 12); return clamp(code, 0, 255); } std::uint8_t compute_frontend_gain_wolfson_gl841(float value, float target_value) { // this code path is similar to what generic wolfson code path uses and uses similar constants, // but is likely incorrect. float inv_gain = target_value / value; inv_gain *= 0.69f; int code = static_cast(283 - 208 / inv_gain); return clamp(code, 0, 255); } std::uint8_t compute_frontend_gain_wolfson_gl846_gl847_gl124(float value, float target_value) { // this code path is similar to what generic wolfson code path uses and uses similar constants, // but is likely incorrect. float inv_gain = target_value / value; int code = static_cast(283 - 208 / inv_gain); return clamp(code, 0, 255); } std::uint8_t compute_frontend_gain_analog_devices(float value, float target_value) { /* The flow of data through the frontend ADC is as follows (see e.g. AD9826 datasheet) input -> apply offset (o = i + 300mV * (OFFSET[8] ? 1 : -1) * (OFFSET[7:0] / 127) -> apply gain (o = i * 6 / (1 + 5 * ( 63 - PGA[5:0] ) / 63 ) ) -> ADC We want to solve the following for {PGA}: {value} = {input} * 6 / (1 + 5 * ( 63 - 0) / 63 ) ) {target_value} = {input} * 6 / (1 + 5 * ( 63 - {PGA}) / 63 ) ) The solution is the following equation: {PGA} = (378 / 5) * ({target_value} - {value} / {target_value}) */ int code = static_cast((378.0f / 5.0f) * ((target_value - value) / target_value)); return clamp(code, 0, 63); } std::uint8_t compute_frontend_gain(float value, float target_value, FrontendType frontend_type) { switch (frontend_type) { case FrontendType::WOLFSON: return compute_frontend_gain_wolfson(value, target_value); case FrontendType::ANALOG_DEVICES: return compute_frontend_gain_analog_devices(value, target_value); case FrontendType::CANON_LIDE_80: return compute_frontend_gain_lide_80(value, target_value); case FrontendType::WOLFSON_GL841: return compute_frontend_gain_wolfson_gl841(value, target_value); case FrontendType::WOLFSON_GL846: case FrontendType::ANALOG_DEVICES_GL847: case FrontendType::WOLFSON_GL124: return compute_frontend_gain_wolfson_gl846_gl847_gl124(value, target_value); default: throw SaneException("Unknown frontend to compute gain for"); } } /** @brief initialize device * Initialize backend and ASIC : registers, motor tables, and gamma tables * then ensure scanner's head is at home. Designed for gl846+ ASICs. * Detects cold boot (ie first boot since device plugged) in this case * an extensice setup up is done at hardware level. * * @param dev device to initialize * @param max_regs umber of maximum used registers */ void sanei_genesys_asic_init(Genesys_Device* dev) { DBG_HELPER(dbg); std::uint8_t val; bool cold = true; // URB 16 control 0xc0 0x0c 0x8e 0x0b len 1 read 0x00 */ dev->interface->get_usb_device().control_msg(REQUEST_TYPE_IN, REQUEST_REGISTER, VALUE_GET_REGISTER, 0x00, 1, &val); DBG (DBG_io2, "%s: value=0x%02x\n", __func__, val); DBG (DBG_info, "%s: device is %s\n", __func__, (val & 0x08) ? "USB 1.0" : "USB2.0"); if (val & 0x08) { dev->usb_mode = 1; } else { dev->usb_mode = 2; } /* Check if the device has already been initialized and powered up. We read register 0x06 and check PWRBIT, if reset scanner has been freshly powered up. This bit will be set to later so that following reads can detect power down/up cycle */ if (!is_testing_mode()) { if (dev->interface->read_register(0x06) & 0x10) { cold = false; } } DBG (DBG_info, "%s: device is %s\n", __func__, cold ? "cold" : "warm"); /* don't do anything if backend is initialized and hardware hasn't been * replug */ if (dev->already_initialized && !cold) { DBG (DBG_info, "%s: already initialized, nothing to do\n", __func__); return; } // set up hardware and registers dev->cmd_set->asic_boot(dev, cold); /* now hardware part is OK, set up device struct */ dev->white_average_data.clear(); dev->dark_average_data.clear(); dev->settings.color_filter = ColorFilter::RED; dev->initial_regs = dev->reg; const auto& sensor = sanei_genesys_find_sensor_any(dev); // Set analog frontend dev->cmd_set->set_fe(dev, sensor, AFE_INIT); dev->already_initialized = true; // Move to home if needed if (dev->model->model_id == ModelId::CANON_8600F) { if (!dev->cmd_set->is_head_home(*dev, ScanHeadId::SECONDARY)) { dev->set_head_pos_unknown(ScanHeadId::SECONDARY); } if (!dev->cmd_set->is_head_home(*dev, ScanHeadId::PRIMARY)) { dev->set_head_pos_unknown(ScanHeadId::SECONDARY); } } dev->cmd_set->move_back_home(dev, true); // Set powersaving (default = 15 minutes) dev->cmd_set->set_powersaving(dev, 15); } void scanner_start_action(Genesys_Device& dev, bool start_motor) { DBG_HELPER(dbg); switch (dev.model->asic_type) { case AsicType::GL646: case AsicType::GL841: case AsicType::GL842: case AsicType::GL843: case AsicType::GL845: case AsicType::GL846: case AsicType::GL847: case AsicType::GL124: break; default: throw SaneException("Unsupported chip"); } if (start_motor) { dev.interface->write_register(0x0f, 0x01); } else { dev.interface->write_register(0x0f, 0); } } void sanei_genesys_set_dpihw(Genesys_Register_Set& regs, unsigned dpihw) { // same across GL646, GL841, GL843, GL846, GL847, GL124 const std::uint8_t REG_0x05_DPIHW_MASK = 0xc0; const std::uint8_t REG_0x05_DPIHW_600 = 0x00; const std::uint8_t REG_0x05_DPIHW_1200 = 0x40; const std::uint8_t REG_0x05_DPIHW_2400 = 0x80; const std::uint8_t REG_0x05_DPIHW_4800 = 0xc0; std::uint8_t dpihw_setting; switch (dpihw) { case 600: dpihw_setting = REG_0x05_DPIHW_600; break; case 1200: dpihw_setting = REG_0x05_DPIHW_1200; break; case 2400: dpihw_setting = REG_0x05_DPIHW_2400; break; case 4800: dpihw_setting = REG_0x05_DPIHW_4800; break; default: throw SaneException("Unknown dpihw value: %d", dpihw); } regs.set8_mask(0x05, dpihw_setting, REG_0x05_DPIHW_MASK); } void regs_set_exposure(AsicType asic_type, Genesys_Register_Set& regs, const SensorExposure& exposure) { switch (asic_type) { case AsicType::GL124: { regs.set24(gl124::REG_EXPR, exposure.red); regs.set24(gl124::REG_EXPG, exposure.green); regs.set24(gl124::REG_EXPB, exposure.blue); break; } case AsicType::GL646: { regs.set16(gl646::REG_EXPR, exposure.red); regs.set16(gl646::REG_EXPG, exposure.green); regs.set16(gl646::REG_EXPB, exposure.blue); break; } case AsicType::GL841: { regs.set16(gl841::REG_EXPR, exposure.red); regs.set16(gl841::REG_EXPG, exposure.green); regs.set16(gl841::REG_EXPB, exposure.blue); break; } case AsicType::GL842: { regs.set16(gl842::REG_EXPR, exposure.red); regs.set16(gl842::REG_EXPG, exposure.green); regs.set16(gl842::REG_EXPB, exposure.blue); break; } case AsicType::GL843: { regs.set16(gl843::REG_EXPR, exposure.red); regs.set16(gl843::REG_EXPG, exposure.green); regs.set16(gl843::REG_EXPB, exposure.blue); break; } case AsicType::GL845: case AsicType::GL846: { regs.set16(gl846::REG_EXPR, exposure.red); regs.set16(gl846::REG_EXPG, exposure.green); regs.set16(gl846::REG_EXPB, exposure.blue); break; } case AsicType::GL847: { regs.set16(gl847::REG_EXPR, exposure.red); regs.set16(gl847::REG_EXPG, exposure.green); regs.set16(gl847::REG_EXPB, exposure.blue); break; } default: throw SaneException("Unsupported asic"); } } void regs_set_optical_off(AsicType asic_type, Genesys_Register_Set& regs) { DBG_HELPER(dbg); switch (asic_type) { case AsicType::GL646: { regs.find_reg(gl646::REG_0x01).value &= ~gl646::REG_0x01_SCAN; break; } case AsicType::GL841: { regs.find_reg(gl841::REG_0x01).value &= ~gl841::REG_0x01_SCAN; break; } case AsicType::GL842: { regs.find_reg(gl842::REG_0x01).value &= ~gl842::REG_0x01_SCAN; break; } case AsicType::GL843: { regs.find_reg(gl843::REG_0x01).value &= ~gl843::REG_0x01_SCAN; break; } case AsicType::GL845: case AsicType::GL846: { regs.find_reg(gl846::REG_0x01).value &= ~gl846::REG_0x01_SCAN; break; } case AsicType::GL847: { regs.find_reg(gl847::REG_0x01).value &= ~gl847::REG_0x01_SCAN; break; } case AsicType::GL124: { regs.find_reg(gl124::REG_0x01).value &= ~gl124::REG_0x01_SCAN; break; } default: throw SaneException("Unsupported asic"); } } bool get_registers_gain4_bit(AsicType asic_type, const Genesys_Register_Set& regs) { switch (asic_type) { case AsicType::GL646: return static_cast(regs.get8(gl646::REG_0x06) & gl646::REG_0x06_GAIN4); case AsicType::GL841: return static_cast(regs.get8(gl841::REG_0x06) & gl841::REG_0x06_GAIN4); case AsicType::GL842: return static_cast(regs.get8(gl842::REG_0x06) & gl842::REG_0x06_GAIN4); case AsicType::GL843: return static_cast(regs.get8(gl843::REG_0x06) & gl843::REG_0x06_GAIN4); case AsicType::GL845: case AsicType::GL846: return static_cast(regs.get8(gl846::REG_0x06) & gl846::REG_0x06_GAIN4); case AsicType::GL847: return static_cast(regs.get8(gl847::REG_0x06) & gl847::REG_0x06_GAIN4); case AsicType::GL124: return static_cast(regs.get8(gl124::REG_0x06) & gl124::REG_0x06_GAIN4); default: throw SaneException("Unsupported chipset"); } } /** * Wait for the scanning head to park */ void sanei_genesys_wait_for_home(Genesys_Device* dev) { DBG_HELPER(dbg); /* clear the parking status whatever the outcome of the function */ dev->parking = false; if (is_testing_mode()) { return; } // read initial status, if head isn't at home and motor is on we are parking, so we wait. // gl847/gl124 need 2 reads for reliable results auto status = scanner_read_status(*dev); dev->interface->sleep_ms(10); status = scanner_read_status(*dev); if (status.is_at_home) { DBG (DBG_info, "%s: already at home\n", __func__); return; } unsigned timeout_ms = 200000; unsigned elapsed_ms = 0; do { dev->interface->sleep_ms(100); elapsed_ms += 100; status = scanner_read_status(*dev); } while (elapsed_ms < timeout_ms && !status.is_at_home); /* if after the timeout, head is still not parked, error out */ if (elapsed_ms >= timeout_ms && !status.is_at_home) { DBG (DBG_error, "%s: failed to reach park position in %dseconds\n", __func__, timeout_ms / 1000); throw SaneException(SANE_STATUS_IO_ERROR, "failed to reach park position"); } } const MotorProfile* get_motor_profile_ptr(const std::vector& profiles, unsigned exposure, const ScanSession& session) { int best_i = -1; for (unsigned i = 0; i < profiles.size(); ++i) { const auto& profile = profiles[i]; if (!profile.resolutions.matches(session.params.yres)) { continue; } if (!profile.scan_methods.matches(session.params.scan_method)) { continue; } if (profile.max_exposure == exposure) { return &profile; } if (profile.max_exposure == 0 || profile.max_exposure >= exposure) { if (best_i < 0) { // no match found yet best_i = i; } else { // test for better match if (profiles[i].max_exposure < profiles[best_i].max_exposure) { best_i = i; } } } } if (best_i < 0) { return nullptr; } return &profiles[best_i]; } const MotorProfile& get_motor_profile(const std::vector& profiles, unsigned exposure, const ScanSession& session) { const auto* profile = get_motor_profile_ptr(profiles, exposure, session); if (profile == nullptr) { throw SaneException("Motor slope is not configured"); } return *profile; } MotorSlopeTable create_slope_table(AsicType asic_type, const Genesys_Motor& motor, unsigned ydpi, unsigned exposure, unsigned step_multiplier, const MotorProfile& motor_profile) { unsigned target_speed_w = ((exposure * ydpi) / motor.base_ydpi); auto table = create_slope_table_for_speed(motor_profile.slope, target_speed_w, motor_profile.step_type, step_multiplier, 2 * step_multiplier, get_slope_table_max_size(asic_type)); return table; } MotorSlopeTable create_slope_table_fastest(AsicType asic_type, unsigned step_multiplier, const MotorProfile& motor_profile) { return create_slope_table_for_speed(motor_profile.slope, motor_profile.slope.max_speed_w, motor_profile.step_type, step_multiplier, 2 * step_multiplier, get_slope_table_max_size(asic_type)); } /** @brief returns the lowest possible ydpi for the device * Parses device entry to find lowest motor dpi. * @param dev device description * @return lowest motor resolution */ int sanei_genesys_get_lowest_ydpi(Genesys_Device *dev) { const auto& resolution_settings = dev->model->get_resolution_settings(dev->settings.scan_method); return resolution_settings.get_min_resolution_y(); } /** @brief returns the lowest possible dpi for the device * Parses device entry to find lowest motor or sensor dpi. * @param dev device description * @return lowest motor resolution */ int sanei_genesys_get_lowest_dpi(Genesys_Device *dev) { const auto& resolution_settings = dev->model->get_resolution_settings(dev->settings.scan_method); return std::min(resolution_settings.get_min_resolution_x(), resolution_settings.get_min_resolution_y()); } /** @brief check is a cache entry may be used * Compares current settings with the cache entry and return * true if they are compatible. * A calibration cache is compatible if color mode and x dpi match the user * requested scan. In the case of CIS scanners, dpi isn't a criteria. * flatbed cache entries are considered too old and then expires if they * are older than the expiration time option, forcing calibration at least once * then given time. */ bool sanei_genesys_is_compatible_calibration(Genesys_Device* dev, const ScanSession& session, const Genesys_Calibration_Cache* cache, bool for_overwrite) { DBG_HELPER(dbg); #ifdef HAVE_SYS_TIME_H struct timeval time; #endif bool compatible = true; const auto& dev_params = session.params; if (dev_params.scan_method != cache->params.scan_method) { dbg.vlog(DBG_io, "incompatible: scan_method %d vs. %d\n", static_cast(dev_params.scan_method), static_cast(cache->params.scan_method)); compatible = false; } if (dev_params.xres != cache->params.xres) { dbg.vlog(DBG_io, "incompatible: params.xres %d vs. %d\n", dev_params.xres, cache->params.xres); compatible = false; } if (dev_params.yres != cache->params.yres) { // exposure depends on selected sensor and we select the sensor according to yres dbg.vlog(DBG_io, "incompatible: params.yres %d vs. %d\n", dev_params.yres, cache->params.yres); compatible = false; } if (dev_params.channels != cache->params.channels) { // exposure depends on total number of pixels at least on gl841 dbg.vlog(DBG_io, "incompatible: params.channels %d vs. %d\n", dev_params.channels, cache->params.channels); compatible = false; } if (dev_params.startx != cache->params.startx) { // exposure depends on total number of pixels at least on gl841 dbg.vlog(DBG_io, "incompatible: params.startx %d vs. %d\n", dev_params.startx, cache->params.startx); compatible = false; } if (dev_params.pixels != cache->params.pixels) { // exposure depends on total number of pixels at least on gl841 dbg.vlog(DBG_io, "incompatible: params.pixels %d vs. %d\n", dev_params.pixels, cache->params.pixels); compatible = false; } if (!compatible) { DBG (DBG_proc, "%s: completed, non compatible cache\n", __func__); return false; } /* a cache entry expires after after expiration time for non sheetfed scanners */ /* this is not taken into account when overwriting cache entries */ #ifdef HAVE_SYS_TIME_H if (!for_overwrite && dev->settings.expiration_time >=0) { gettimeofday(&time, nullptr); if ((time.tv_sec - cache->last_calibration > dev->settings.expiration_time*60) && !dev->model->is_sheetfed && (dev->settings.scan_method == ScanMethod::FLATBED)) { DBG (DBG_proc, "%s: expired entry, non compatible cache\n", __func__); return false; } } #endif return true; } /** @brief build lookup table for digital enhancements * Function to build a lookup table (LUT), often used by scanners to implement brightness/contrast/gamma or by backends to speed binarization/thresholding offset and slope inputs are -127 to +127 slope rotates line around central input/output val, 0 makes horizontal line pos zero neg . x . . x . x . . x out . x .xxxxxxxxxxx . x . x . . x ....x....... ............ .......x.... in in in offset moves line vertically, and clamps to output range 0 keeps the line crossing the center of the table high low . xxxxxxxx . . x . out x . x . . x ............ xxxxxxxx.... in in out_min/max provide bounds on output values, useful when building thresholding lut. 0 and 255 are good defaults otherwise. * @param lut pointer where to store the generated lut * @param in_bits number of bits for in values * @param out_bits number of bits of out values * @param out_min minimal out value * @param out_max maximal out value * @param slope slope of the generated data * @param offset offset of the generated data */ void sanei_genesys_load_lut(unsigned char* lut, int in_bits, int out_bits, int out_min, int out_max, int slope, int offset) { DBG_HELPER(dbg); int i, j; double shift, rise; int max_in_val = (1 << in_bits) - 1; int max_out_val = (1 << out_bits) - 1; std::uint8_t* lut_p8 = lut; std::uint16_t* lut_p16 = reinterpret_cast(lut); /* slope is converted to rise per unit run: * first [-127,127] to [-.999,.999] * then to [-PI/4,PI/4] then [0,PI/2] * then take the tangent (T.O.A) * then multiply by the normal linear slope * because the table may not be square, i.e. 1024x256*/ auto pi_4 = M_PI / 4.0; rise = std::tan(static_cast(slope) / 128 * pi_4 + pi_4) * max_out_val / max_in_val; /* line must stay vertically centered, so figure * out vertical offset at central input value */ shift = static_cast(max_out_val) / 2 - (rise * max_in_val / 2); /* convert the user offset setting to scale of output * first [-127,127] to [-1,1] * then to [-max_out_val/2,max_out_val/2]*/ shift += static_cast(offset) / 127 * max_out_val / 2; for (i = 0; i <= max_in_val; i++) { j = static_cast(rise * i + shift); /* cap data to required range */ if (j < out_min) { j = out_min; } else if (j > out_max) { j = out_max; } /* copy result according to bit depth */ if (out_bits <= 8) { *lut_p8 = j; lut_p8++; } else { *lut_p16 = j; lut_p16++; } } } } // namespace genesys