#include "pump.hpp" #include #include using namespace sim::coolant; pump::pump(fluid_holder* src, fluid_holder* dst, double mass, double radius, double power, double l_per_rev, double friction, mode_t mode, double target) : src(src), dst(dst), mass(mass), radius(radius), l_per_rev(l_per_rev), friction(friction), max_power(power), mode(mode), target(target) { } double pump::get_flow_target() const { return l_per_rev * get_rpm() * 60; } double pump::get_flow() const { return flow; } double pump::get_flow_mass() const { return src->fluid.l_to_g(flow); } double pump::get_rpm() const { return velocity / (M_PI * mass * 0.001 * radius * radius); } double pump::get_power() const { return power; } const char* pump::get_state_string() { if(!powered) return "Off"; if(power == 0) return "Idle"; return "On"; } static double calc_work(double j, double mass) { double m = 1; if(j < 0) { m = -1; } return m * std::sqrt(m * j / (mass * 0.001)); } void pump::update(double dt) { if(powered) { switch(mode) { case mode_t::SRC: power = pid.calculate(dt, target, src->get_gas_volume()); break; case mode_t::DST: power = pid.calculate(dt, target, dst->get_level()); break; case mode_t::NONE: power = 1; break; } velocity += calc_work(dt * power * max_power, mass); } else { power = 0; } double src_heat = src->get_heat(); double p_diff_1 = dst->get_pressure() - src->get_pressure(); double max_volume = ignore_dst_level ? src->get_level() : std::min(src->get_level(), dst->get_volume() - dst->get_level()); double src_volume = src->extract_fluid(std::min(get_flow_target() * dt, max_volume)); double dst_volume = dst->add_fluid(src_volume, src_heat); double p_diff_2 = dst->get_pressure() - src->get_pressure(); double p_diff = (p_diff_1 + p_diff_2) / 2; double work = p_diff * dst_volume * 0.001 + get_rpm() * 60 * dt * friction; velocity = std::max(velocity - calc_work(work, mass), 0.0); flow = dst_volume / dt; }