/* // Copyright (c) 2018 Intel Corporation // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. */ #include "stepwisecontroller.hpp" #include "ec/stepwise.hpp" #include "errors/exception.hpp" #include "util.hpp" #include "zone.hpp" #include #include #include #include #include #include #include #include namespace pid_control { void StepwiseController::process(void) { // Get input value double input = inputProc(); ec::StepwiseInfo info = getStepwiseInfo(); double output = lastOutput; // Calculate new output if hysteresis allows if (std::isnan(output)) { output = ec::stepwise(info, input); lastInput = input; } else if ((input - lastInput) > info.positiveHysteresis) { output = ec::stepwise(info, input); lastInput = input; } else if ((lastInput - input) > info.negativeHysteresis) { output = ec::stepwise(info, input); lastInput = input; } lastOutput = output; // Output new value outputProc(output); return; } std::unique_ptr StepwiseController::createStepwiseController( ZoneInterface* owner, const std::string& id, const std::vector& inputs, const ec::StepwiseInfo& initial) { // StepwiseController requires at least 1 input if (inputs.empty()) { throw ControllerBuildException("Stepwise controller missing inputs"); return nullptr; } auto thermal = std::make_unique(id, inputs, owner); thermal->setStepwiseInfo(initial); return thermal; } double StepwiseController::inputProc(void) { double value = std::numeric_limits::lowest(); for (const auto& in : _inputs) { value = std::max(value, _owner->getCachedValue(in)); } return value; } void StepwiseController::outputProc(double value) { if (getStepwiseInfo().isCeiling) { _owner->addRPMCeiling(value); } else { _owner->addSetPoint(value, _id); } return; } } // namespace pid_control