sane-project-backends/backend/genesys/motor.cpp

193 wiersze
6.6 KiB
C++

/* 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/>.
*/
#define DEBUG_DECLARE_ONLY
#include "low.h"
#include "motor.h"
#include "utilities.h"
#include <cmath>
#include <numeric>
namespace genesys {
unsigned MotorSlope::get_table_step_shifted(unsigned step, StepType step_type) const
{
// first two steps are always equal to the initial speed
if (step < 2) {
return initial_speed_w >> static_cast<unsigned>(step_type);
}
step--;
float initial_speed_v = 1.0f / initial_speed_w;
float speed_v = std::sqrt(initial_speed_v * initial_speed_v + 2 * acceleration * step);
return static_cast<unsigned>(1.0f / speed_v) >> static_cast<unsigned>(step_type);
}
float compute_acceleration_for_steps(unsigned initial_w, unsigned max_w, unsigned steps)
{
float initial_speed_v = 1.0f / static_cast<float>(initial_w);
float max_speed_v = 1.0f / static_cast<float>(max_w);
return (max_speed_v * max_speed_v - initial_speed_v * initial_speed_v) / (2 * steps);
}
MotorSlope MotorSlope::create_from_steps(unsigned initial_w, unsigned max_w,
unsigned steps)
{
MotorSlope slope;
slope.initial_speed_w = initial_w;
slope.max_speed_w = max_w;
slope.acceleration = compute_acceleration_for_steps(initial_w, max_w, steps);
return slope;
}
void MotorSlopeTable::slice_steps(unsigned count, unsigned step_multiplier)
{
if (count > table.size() || count < step_multiplier) {
throw SaneException("Invalid steps count");
}
count = align_multiple_floor(count, step_multiplier);
table.resize(count);
generate_pixeltime_sum();
}
void MotorSlopeTable::expand_table(unsigned count, unsigned step_multiplier)
{
if (table.empty()) {
throw SaneException("Can't expand empty table");
}
count = align_multiple_ceil(count, step_multiplier);
table.resize(table.size() + count, table.back());
generate_pixeltime_sum();
}
void MotorSlopeTable::generate_pixeltime_sum()
{
pixeltime_sum_ = std::accumulate(table.begin(), table.end(),
std::size_t{0}, std::plus<std::size_t>());
}
unsigned get_slope_table_max_size(AsicType asic_type)
{
switch (asic_type) {
case AsicType::GL646:
case AsicType::GL841:
case AsicType::GL842: return 255;
case AsicType::GL843:
case AsicType::GL845:
case AsicType::GL846:
case AsicType::GL847:
case AsicType::GL124: return 1024;
default:
throw SaneException("Unknown asic type");
}
}
MotorSlopeTable create_slope_table_for_speed(const MotorSlope& slope, unsigned target_speed_w,
StepType step_type, unsigned steps_alignment,
unsigned min_size, unsigned max_size)
{
DBG_HELPER_ARGS(dbg, "target_speed_w: %d, step_type: %d, steps_alignment: %d, min_size: %d",
target_speed_w, static_cast<unsigned>(step_type), steps_alignment, min_size);
MotorSlopeTable table;
unsigned step_shift = static_cast<unsigned>(step_type);
unsigned target_speed_shifted_w = target_speed_w >> step_shift;
unsigned max_speed_shifted_w = slope.max_speed_w >> step_shift;
if (target_speed_shifted_w < max_speed_shifted_w) {
dbg.vlog(DBG_warn, "failed to reach target speed %d %d", target_speed_w,
slope.max_speed_w);
}
if (target_speed_shifted_w >= std::numeric_limits<std::uint16_t>::max()) {
throw SaneException("Target motor speed is too low");
}
unsigned final_speed = std::max(target_speed_shifted_w, max_speed_shifted_w);
table.table.reserve(max_size);
while (table.table.size() < max_size - 1) {
unsigned current = slope.get_table_step_shifted(table.table.size(), step_type);
if (current <= final_speed) {
break;
}
table.table.push_back(current);
}
// make sure the target speed (or the max speed if target speed is too high) is present in
// the table
table.table.push_back(final_speed);
// fill the table up to the specified size
while (table.table.size() < max_size - 1 &&
(table.table.size() % steps_alignment != 0 || table.table.size() < min_size))
{
table.table.push_back(table.table.back());
}
table.generate_pixeltime_sum();
return table;
}
std::ostream& operator<<(std::ostream& out, const MotorSlope& slope)
{
out << "MotorSlope{\n"
<< " initial_speed_w: " << slope.initial_speed_w << '\n'
<< " max_speed_w: " << slope.max_speed_w << '\n'
<< " a: " << slope.acceleration << '\n'
<< '}';
return out;
}
std::ostream& operator<<(std::ostream& out, const MotorProfile& profile)
{
out << "MotorProfile{\n"
<< " max_exposure: " << profile.max_exposure << '\n'
<< " step_type: " << profile.step_type << '\n'
<< " motor_vref: " << profile.motor_vref << '\n'
<< " resolutions: " << format_indent_braced_list(4, profile.resolutions) << '\n'
<< " scan_methods: " << format_indent_braced_list(4, profile.scan_methods) << '\n'
<< " slope: " << format_indent_braced_list(4, profile.slope) << '\n'
<< '}';
return out;
}
std::ostream& operator<<(std::ostream& out, const Genesys_Motor& motor)
{
out << "Genesys_Motor{\n"
<< " id: " << motor.id << '\n'
<< " base_ydpi: " << motor.base_ydpi << '\n'
<< " profiles: "
<< format_indent_braced_list(4, format_vector_indent_braced(4, "MotorProfile",
motor.profiles)) << '\n'
<< " fast_profiles: "
<< format_indent_braced_list(4, format_vector_indent_braced(4, "MotorProfile",
motor.fast_profiles)) << '\n'
<< '}';
return out;
}
} // namespace genesys