1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367
|
/* sane - Scanner Access Now Easy.
Copyright (C) 2019 Povilas Kanapickas <povilas@radix.lt>
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 <https://www.gnu.org/licenses/>.
*/
#ifndef BACKEND_GENESYS_SETTINGS_H
#define BACKEND_GENESYS_SETTINGS_H
#include "enums.h"
#include "serialize.h"
#include "utilities.h"
#include "sensor.h"
namespace genesys {
struct Genesys_Settings
{
ScanMethod scan_method = ScanMethod::FLATBED;
ScanColorMode scan_mode = ScanColorMode::LINEART;
// horizontal dpi
unsigned xres = 0;
// vertical dpi
unsigned yres = 0;
//x start on scan table in mm
float tl_x = 0;
// y start on scan table in mm
float tl_y = 0;
// number of lines at scan resolution
unsigned int lines = 0;
// number of pixels expected from the scanner
unsigned int pixels = 0;
// number of pixels expected by the frontend
unsigned requested_pixels = 0;
// bit depth of the scan
unsigned int depth = 0;
ColorFilter color_filter = ColorFilter::NONE;
// value for contrast enhancement in the [-100..100] range
int contrast = 0;
// value for brightness enhancement in the [-100..100] range
int brightness = 0;
// cache entries expiration time
int expiration_time = 0;
unsigned get_channels() const
{
if (scan_mode == ScanColorMode::COLOR_SINGLE_PASS)
return 3;
return 1;
}
};
std::ostream& operator<<(std::ostream& out, const Genesys_Settings& settings);
struct SetupParams {
static constexpr unsigned NOT_SET = std::numeric_limits<unsigned>::max();
static constexpr unsigned NOT_SET_I = std::numeric_limits<int>::max();
// resolution in x direction
unsigned xres = NOT_SET;
// resolution in y direction
unsigned yres = NOT_SET;
// start pixel in X direction, from dummy_pixel + 1. Counted in terms of xres.
unsigned startx = NOT_SET;
// start pixel in Y direction, counted according to base_ydpi
unsigned starty = NOT_SET;
// the number of pixels in X direction. Counted in terms of xres.
// Note that each logical pixel may correspond to more than one CCD pixel, see CKSEL and
// GenesysSensor::ccd_pixels_per_system_pixel()
unsigned pixels = NOT_SET;
// the number of pixels in the X direction as requested by the frontend. This will be different
// from `pixels` if the X resolution requested by the frontend is different than the actual
// resolution. This is only needed to compute dev->total_bytes_to_read. If 0, then the value
// is the same as pixels.
// TODO: move the computation of total_bytes_to_read to a higher layer.
unsigned requested_pixels = 0;
// the number of pixels in Y direction
unsigned lines = NOT_SET;
// the depth of the scan in bits. Allowed are 1, 8, 16
unsigned depth = NOT_SET;
// the number of channels
unsigned channels = NOT_SET;
ScanMethod scan_method = static_cast<ScanMethod>(NOT_SET);
ScanColorMode scan_mode = static_cast<ScanColorMode>(NOT_SET);
ColorFilter color_filter = static_cast<ColorFilter>(NOT_SET);
// the values for contrast and brightness adjustment in the range of [-100..100]
int contrast_adjustment = NOT_SET_I;
int brightness_adjustment = NOT_SET_I;
ScanFlag flags = ScanFlag::NONE;
unsigned get_requested_pixels() const
{
if (requested_pixels != 0) {
return requested_pixels;
}
return pixels;
}
void assert_valid() const
{
if (xres == NOT_SET || yres == NOT_SET || startx == NOT_SET || starty == NOT_SET ||
pixels == NOT_SET || lines == NOT_SET ||depth == NOT_SET || channels == NOT_SET ||
scan_method == static_cast<ScanMethod>(NOT_SET) ||
scan_mode == static_cast<ScanColorMode>(NOT_SET) ||
color_filter == static_cast<ColorFilter>(NOT_SET) ||
contrast_adjustment == NOT_SET_I || brightness_adjustment == NOT_SET_I)
{
throw std::runtime_error("SetupParams are not valid");
}
}
bool operator==(const SetupParams& other) const
{
return xres == other.xres &&
yres == other.yres &&
startx == other.startx &&
starty == other.starty &&
pixels == other.pixels &&
requested_pixels == other.requested_pixels &&
lines == other.lines &&
depth == other.depth &&
channels == other.channels &&
scan_method == other.scan_method &&
scan_mode == other.scan_mode &&
color_filter == other.color_filter &&
contrast_adjustment == other.contrast_adjustment &&
brightness_adjustment == other.brightness_adjustment &&
flags == other.flags;
}
};
std::ostream& operator<<(std::ostream& out, const SetupParams& params);
template<class Stream>
void serialize(Stream& str, SetupParams& x)
{
serialize(str, x.xres);
serialize(str, x.yres);
serialize(str, x.startx);
serialize(str, x.starty);
serialize(str, x.pixels);
serialize(str, x.requested_pixels);
serialize(str, x.lines);
serialize(str, x.depth);
serialize(str, x.channels);
serialize(str, x.scan_method);
serialize(str, x.scan_mode);
serialize(str, x.color_filter);
serialize(str, x.contrast_adjustment);
serialize(str, x.brightness_adjustment);
serialize(str, x.flags);
}
struct ScanSession {
SetupParams params;
// whether the session setup has been computed via compute_session()
bool computed = false;
// specifies the full resolution of the sensor that is being used.
unsigned full_resolution = 0;
// the optical resolution of the sensor that is being used.
unsigned optical_resolution = 0;
// the number of pixels at the optical resolution, not including segmentation overhead.
unsigned optical_pixels = 0;
// the number of pixels at the optical resolution, including segmentation overhead.
// only on gl846, g847
unsigned optical_pixels_raw = 0;
// the number of optical scan lines. Equal to output_line_count on CCD scanners.
unsigned optical_line_count = 0;
// the resolution of the output data.
unsigned output_resolution = 0;
// the offset in pixels from the beginning of output data
unsigned output_startx = 0;
// the number of pixels in output data (after desegmentation)
unsigned output_pixels = 0;
// the number of bytes in the output of a channel of a single line (after desegmentation)
unsigned output_channel_bytes = 0;
// the number of bytes in the output of a single line (after desegmentation)
unsigned output_line_bytes = 0;
// the number of bytes per line in the output data from the scanner (before desegmentation)
// Equal to output_line_bytes if sensor does not have segments
unsigned output_line_bytes_raw = 0;
// the number of bytes per line as requested by the frontend
unsigned output_line_bytes_requested = 0;
// the number of lines in the output of the scanner. This must be larger than the user
// requested number due to line staggering and color channel shifting.
unsigned output_line_count = 0;
// the total number of bytes to read from the scanner (before desegmentation)
unsigned output_total_bytes_raw = 0;
// the total number of bytes to read from the scanner (after desegmentation)
unsigned output_total_bytes = 0;
// the number of staggered lines (i.e. lines that overlap during scanning due to line being
// thinner than the CCD element). Computed according to stagger_y.
unsigned num_staggered_lines = 0;
// the number of lines that color channels shift due to different physical positions of
// different color channels.
unsigned max_color_shift_lines = 0;
// actual line shift of the red color
unsigned color_shift_lines_r = 0;
// actual line shift of the green color
unsigned color_shift_lines_g = 0;
// actual line shift of the blue color
unsigned color_shift_lines_b = 0;
// The shifts that need to be applied to the output pixels in x direction.
StaggerConfig stagger_x;
// The shifts that need to be applied to the output pixels in y direction.
StaggerConfig stagger_y;
// the number of scanner segments used in the current scan
unsigned segment_count = 1;
// the physical pixel positions that are sent to the registers
unsigned pixel_startx = 0;
unsigned pixel_endx = 0;
/* The following defines the ratio between logical pixel count and pixel count setting sent to
the scanner. The ratio is affected by the following:
- Certain scanners just like to multiply the pixel number by a multiplier that depends on
the resolution.
- The sensor may be configured to output one value per multiple physical pixels
- The scanner will automatically average the pixels that come from the sensor using a
certain ratio.
*/
Ratio pixel_count_ratio = Ratio{1, 1};
// Distance in pixels between consecutive pixels, e.g. between odd and even pixels. Note that
// the number of segments can be large.
// only on gl124, gl846, gl847
unsigned conseq_pixel_dist = 0;
// The number of "even" pixels to scan. This corresponds to the number of pixels that will be
// scanned from a single segment
// only on gl124, gl846, gl847
unsigned output_segment_pixel_group_count = 0;
// The number of bytes to skip at start of line during desegmentation.
// Currently it's always zero.
unsigned output_segment_start_offset = 0;
// How many pixels the shading data is offset to the right from the acquired data. Calculated
// in shading resolution.
int shading_pixel_offset = 0;
// the size of the read buffer.
size_t buffer_size_read = 0;
// whether to enable ledadd functionality
bool enable_ledadd = false;
// whether calibration should be performed host-side
bool use_host_side_calib = false;
// whether gray scanning should be performed host-side (scan as color and merge to gray)
bool use_host_side_gray = false;
void assert_computed() const
{
if (!computed) {
throw std::runtime_error("ScanSession is not computed");
}
}
bool operator==(const ScanSession& other) const;
};
std::ostream& operator<<(std::ostream& out, const ScanSession& session);
template<class Stream>
void serialize(Stream& str, ScanSession& x)
{
serialize(str, x.params);
serialize_newline(str);
serialize(str, x.computed);
serialize(str, x.full_resolution);
serialize(str, x.optical_resolution);
serialize(str, x.optical_pixels);
serialize(str, x.optical_pixels_raw);
serialize(str, x.optical_line_count);
serialize(str, x.output_resolution);
serialize(str, x.output_startx);
serialize(str, x.output_pixels);
serialize(str, x.output_channel_bytes);
serialize(str, x.output_line_bytes);
serialize(str, x.output_line_bytes_raw);
serialize(str, x.output_line_bytes_requested);
serialize(str, x.output_line_count);
serialize(str, x.output_total_bytes_raw);
serialize(str, x.output_total_bytes);
serialize(str, x.num_staggered_lines);
serialize(str, x.max_color_shift_lines);
serialize(str, x.color_shift_lines_r);
serialize(str, x.color_shift_lines_g);
serialize(str, x.color_shift_lines_b);
serialize(str, x.stagger_x);
serialize(str, x.stagger_y);
serialize(str, x.segment_count);
serialize(str, x.pixel_startx);
serialize(str, x.pixel_endx);
serialize(str, x.pixel_count_ratio);
serialize(str, x.conseq_pixel_dist);
serialize(str, x.output_segment_pixel_group_count);
serialize(str, x.output_segment_start_offset);
serialize(str, x.shading_pixel_offset);
serialize(str, x.buffer_size_read);
serialize(str, x.enable_ledadd);
serialize(str, x.use_host_side_calib);
serialize(str, x.use_host_side_gray);
}
std::ostream& operator<<(std::ostream& out, const SANE_Parameters& params);
} // namespace genesys
#endif // BACKEND_GENESYS_SETTINGS_H
|