eis/eqpalg/utility/expression_engine.cpp

255 lines
7.5 KiB
C++
Raw Normal View History

// eqpalg/utility/expression_engine.cpp
#include <eqpalg/utility/expression_engine.h>
#include <glob/SingletonTemplate.h>
#include <eqpalg/gb_item_memory.h>
#include <mix_cc/type/mix_time.h>
#include <log4cplus/LOG.h>
ExpressionEngine::ExpressionEngine(std::map<std::string, double>& mm_vars,
std::vector<std::string>& m_tags)
: mm_vars_(mm_vars), m_tags_(m_tags) {
// 预初始化变量,等效于原 ExpModule::init()
// 确保 mm_vars 中有 tagN/pN/pvN/now 等键,防止后续表达式求值时变量缺失
if (!m_tags_.empty()) {
ensureVarCache(var_cache_, m_tags_.size());
auto now = std::chrono::system_clock::now();
for (size_t i = 0; i < m_tags_.size(); i++) {
double value = 0.0;
try {
value = SingletonTemplate<GlobaltemSharedMemory>::GetInstance()[m_tags_[i]];
} catch (...) {
value = 0.0;
}
mm_vars_[var_cache_.p_keys[i]] = value;
mm_vars_[var_cache_.tag_keys[i]] = value;
for (size_t j = 0; j < PV_NUM; j++) {
mm_vars_[var_cache_.pv_keys[i][j]] = value;
}
}
mm_vars_["now"] = std::chrono::duration_cast<std::chrono::milliseconds>(
now.time_since_epoch())
.count();
mm_vars_["stime"] = 0;
mm_vars_["etime"] = 0;
mm_vars_["time"] = 0;
}
}
// 惰性初始化 VarsCache首次变量刷新时调用
static void ensureVarCache(VarsCache& vc, size_t tag_count) {
if (vc.tag_num == 0 && tag_count > 0) {
vc.init(tag_count, 6);
}
}
int ExpressionEngine::registerExpression(const std::string& name,
const std::string& raw_exp_str) {
if (exps_.count(name)) return 0; // 已注册,幂等
auto& entry = exps_[name];
entry.raw_str = raw_exp_str;
// 通过 FunVars 替换状态函数KeepC/KeepT/RiseEdge/Detect为 funN 变量
auto [ok, processed] = fun_vars_.add_exp_str(raw_exp_str, &mm_vars_);
if (!ok) return -1;
entry.processed_str = processed;
// 编译表达式
try {
entry.ptr = std::make_unique<MExp>(processed, &mm_vars_);
} catch (const std::exception& e) {
return -1;
}
// 从表达式中提取 hold(n,T) 模式
initHoldExpStr(processed);
return 0;
}
int ExpressionEngine::registerRawExpression(const std::string& name,
const std::string& raw_exp_str) {
if (exps_.count(name)) return 0;
auto& entry = exps_[name];
entry.raw_str = raw_exp_str;
entry.processed_str = raw_exp_str;
try {
entry.ptr = std::make_unique<MExp>(raw_exp_str, &mm_vars_);
} catch (const std::exception& e) {
return -1;
}
return 0;
}
double ExpressionEngine::evaluate(const std::string& name) {
auto it = exps_.find(name);
if (it == exps_.end()) return 0.0;
return it->second.ptr->evaluate();
}
bool ExpressionEngine::evaluateBool(const std::string& name) {
return static_cast<bool>(evaluate(name));
}
int ExpressionEngine::initHoldExpStr(const std::string& exp_str) {
auto hold_sub_strs = HoldTime::find_substr(exp_str, "_HE");
for (const auto& sub_str : hold_sub_strs) {
bool flag;
double timeM;
std::string tagi;
std::string var_name;
std::tie(flag, timeM, tagi, var_name) = HoldTime::find_hold(sub_str);
if (flag) {
if (hold_times_.find(var_name) == hold_times_.end()) {
hold_times_.emplace(std::make_pair(
var_name, std::make_unique<HoldTime>(timeM, tagi, var_name)));
}
}
}
refreshHoldVars();
return 0;
}
// ========== FunVars 控制 ==========
void ExpressionEngine::autoResetFunVars() {
if (fun_vars_need_reset_) {
fun_vars_.refresh_fun_vars(true, &mm_vars_);
fun_vars_need_reset_ = false;
}
}
void ExpressionEngine::markFunVarsNeedReset() {
fun_vars_need_reset_ = true;
}
void ExpressionEngine::forceResetFunVars() {
fun_vars_.refresh_fun_vars(true, &mm_vars_);
fun_vars_need_reset_ = false;
}
void ExpressionEngine::refreshFromMemory(
const TimePoint& now_time,
mix_cc::time_range_t& query_time_range) {
ensureVarCache(var_cache_, m_tags_.size());
for (size_t i = 0; i < m_tags_.size(); i++) {
double current =
SingletonTemplate<GlobaltemSharedMemory>::GetInstance()[m_tags_[i]];
// pN = 旧 tagN
mm_vars_[var_cache_.p_keys[i]] = mm_vars_[var_cache_.tag_keys[i]];
// tagN = 当前值
mm_vars_[var_cache_.tag_keys[i]] = current;
// pvN 历史移位
for (size_t j = PV_NUM - 1; j > 0; j--) {
mm_vars_[var_cache_.pv_keys[i][j]] =
mm_vars_[var_cache_.pv_keys[i][j - 1]];
}
mm_vars_[var_cache_.pv_keys[i][0]] = current;
}
mm_vars_["now"] = std::chrono::duration_cast<std::chrono::milliseconds>(
now_time.time_since_epoch())
.count();
fun_vars_.refresh_fun_vars(false, &mm_vars_);
query_time_range.set_right(now_time);
refreshHoldVars();
}
void ExpressionEngine::refreshFromIhdRow(
int row,
const Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic>& queried_data,
const std::vector<TimePoint>& queried_time,
TimePoint& now_time,
mix_cc::time_range_t& query_time_range) {
if (queried_data.rows() == 0 || queried_data.cols() == 0) return;
if (row < 0 || row >= queried_data.rows()) return;
if (static_cast<int>(m_tags_.size()) > queried_data.cols()) return;
ensureVarCache(var_cache_, m_tags_.size());
for (size_t i = 0; i < m_tags_.size(); i++) {
// pN = 旧 tagN
mm_vars_[var_cache_.p_keys[i]] = mm_vars_[var_cache_.tag_keys[i]];
// tagN = 当前行数据
mm_vars_[var_cache_.tag_keys[i]] = queried_data(row, i);
// pvN 历史移位
for (size_t j = PV_NUM - 1; j > 0; j--) {
mm_vars_[var_cache_.pv_keys[i][j]] =
mm_vars_[var_cache_.pv_keys[i][j - 1]];
}
mm_vars_[var_cache_.pv_keys[i][0]] =
mm_vars_[var_cache_.tag_keys[i]];
}
mm_vars_["now"] =
mix_cc::mix_time_t(queried_time[row]).to_milliseconds();
now_time = queried_time[row];
query_time_range.set_right(queried_time[row]);
fun_vars_.refresh_fun_vars(false, &mm_vars_);
refreshHoldVars();
}
void ExpressionEngine::refreshHoldVars() {
for (auto& kv : hold_times_) {
mm_vars_[kv.first] =
static_cast<double>(kv.second->update_value(mm_vars_[kv.second->tagi]));
}
}
// ========== 调试 ==========
void ExpressionEngine::printVars(const std::string& exp_str) {
// 简单调试输出:遍历所有变量
// 调用方可使用 LOG 宏自行格式化
}
int ExpressionEngine::firstFill(int data_source, TimePoint& now_time,
mix_cc::time_range_t& query_time_range) {
// 惰性初始化 VarsCache
ensureVarCache(var_cache_, m_tags_.size());
for (size_t i = 0; i < m_tags_.size(); i++) {
auto idx = std::to_string(i + 1);
double value = 0.0;
// 从共享内存读取当前值
value = SingletonTemplate<GlobaltemSharedMemory>::GetInstance()[m_tags_[i]];
mm_vars_["s" + idx] = value;
mm_vars_["p" + idx] = value;
mm_vars_["tag" + idx] = value;
mm_vars_["mv2_tag" + idx] = 0;
mm_vars_["mv2_p" + idx] = 0;
mm_vars_["up_tag" + idx] = 0;
mm_vars_["dw_tag" + idx] = 0;
mm_vars_["mx_tag" + idx] = value;
mm_vars_["mi_tag" + idx] = value;
auto pv_str = "pv" + idx;
mm_vars_[pv_str + "_0"] = value;
mm_vars_[pv_str + "_1"] = value;
mm_vars_[pv_str + "_2"] = value;
mm_vars_[pv_str + "_3"] = value;
mm_vars_[pv_str + "_4"] = value;
mm_vars_[pv_str + "_5"] = value;
}
mm_vars_["stime"] = 0;
mm_vars_["now"] = 0;
mm_vars_["etime"] = 0;
mm_vars_["time"] = 0;
return 0;
}