34#include "simgear/misc/strutils.hxx"
36#include "FGFunction.h"
38#include "FGRealValue.h"
39#include "input_output/FGXMLElement.h"
40#include "math/FGFunctionValue.h"
41#include "input_output/string_utilities.h"
52const double invlog2val = 1.0/log10(2.0);
53constexpr unsigned int MaxArgs = 9999;
63 size_t NumberOfArguments(
void)
const {
return Parameters.size(); }
64 FGParameter* FirstParameter(
void)
const {
return *(Parameters.cbegin()); }
65 const Element* GetElement(
void)
const {
return element; }
68 const vector<FGParameter_ptr> Parameters;
69 const Element_ptr element;
74template<
typename func_t,
unsigned int Nmin>
80 FGFunction::OddEven odd_even=FGFunction::OddEven::Either)
83 Load(el, v, fdmex, prefix);
84 CheckMinArguments(el, Nmin);
85 CheckMaxArguments(el, Nmax);
86 CheckOddOrEvenArguments(el, odd_even);
89 double GetValue(
void)
const override {
90 return cached ? cachedValue : f(Parameters);
94 void bind(
Element* el,
const string& Prefix)
override {
95 string nName = CreateOutputNode(el, Prefix);
107template<
typename func_t>
111 aFunc(
const func_t& _f, std::shared_ptr<FGPropertyManager> pm,
Element* el,
112 const string& Prefix)
116 ostringstream buffer;
118 <<
"<" << el->
GetName() <<
"> should have no arguments." <<
reset
126 double GetValue(
void)
const override {
127 double result = cached ? cachedValue : f();
128 if (pNode) pNode->setDoubleValue(result);
141 void bind(
Element* el,
const string& Prefix)
override {
142 CreateOutputNode(el, Prefix);
144 if (pNode) pNode->setDoubleValue(f());
153bool GetBinary(
double val,
const string &ctxMsg)
156 if (val < 1E-9)
return false;
157 else if (val-1 < 1E-9)
return true;
160 <<
"Malformed conditional check in function definition."
170FGFunction* make_MathFn(
double(*math_fn)(
double), FGFDMExec* fdmex, Element* el,
171 const string& prefix, FGPropertyValue* v)
173 auto f = [math_fn](
const std::vector<FGParameter_ptr> &p)->
double {
174 return math_fn(p[0]->GetValue());
176 return new aFunc<decltype(f), 1>(f, fdmex, el, prefix, v);
184template<
typename func_t>
185FGParameter_ptr VarArgsFn(
const func_t& _f, FGFDMExec* fdmex, Element* el,
186 const string& prefix, FGPropertyValue* v)
189 return new aFunc<func_t, 2>(_f, fdmex, el, prefix, v, MaxArgs);
191 catch(WrongNumberOfArguments& e) {
192 if ((e.GetElement() == el) && (e.NumberOfArguments() == 1)) {
194 <<
"<" << el->GetName()
195 <<
"> only has one argument which makes it a no-op." << endl
196 <<
"Its argument will be evaluated but <" << el->GetName()
198 return e.FirstParameter();
211 Load(el, var, fdmex, prefix);
212 CheckMinArguments(el, 1);
213 CheckMaxArguments(el, 1);
217 if (!sCopyTo.empty()) {
218 if (sCopyTo.find(
"#") != string::npos) {
219 if (is_number(prefix))
220 sCopyTo = replace(sCopyTo,
"#",prefix);
223 <<
"Illegal use of the special character '#'" <<
reset << endl
224 <<
"The 'copyto' argument in function " << Name <<
" is ignored."
230 pCopyTo = PropertyManager->GetNode(sCopyTo);
233 <<
"Property \"" << sCopyTo
234 <<
"\" must be previously defined in function " << Name <<
reset
235 <<
"The 'copyto' argument is ignored." << endl;
241void FGFunction::CheckMinArguments(
Element* el,
unsigned int _min)
243 if (Parameters.size() < _min) {
244 ostringstream buffer;
246 <<
"<" << el->
GetName() <<
"> should have at least " << _min
247 <<
" argument(s)." <<
reset << endl;
254void FGFunction::CheckMaxArguments(Element* el,
unsigned int _max)
256 if (Parameters.size() > _max) {
257 ostringstream buffer;
259 <<
"<" << el->GetName() <<
"> should have no more than " << _max
260 <<
" argument(s)." <<
reset << endl;
261 throw WrongNumberOfArguments(buffer.str(), Parameters, el);
267void FGFunction::CheckOddOrEvenArguments(Element* el, OddEven odd_even)
272 if (Parameters.size() % 2 == 1) {
274 <<
"<" << el->GetName() <<
"> must have an even number of arguments."
276 throw BaseException(
"Fatal Error");
280 if (Parameters.size() % 2 == 0) {
282 <<
"<" << el->GetName() <<
"> must have an odd number of arguments."
284 throw BaseException(
"Fatal Error");
294shared_ptr<RandomNumberGenerator> makeRandomGenerator(Element *el, FGFDMExec* fdmex)
296 string seed_attr = el->GetAttributeValue(
"seed");
297 if (seed_attr.empty())
298 return fdmex->GetRandomGenerator();
299 else if (seed_attr ==
"time_now")
300 return make_shared<RandomNumberGenerator>();
302 unsigned int seed = atoi(seed_attr.c_str());
303 return make_shared<RandomNumberGenerator>(seed);
309void FGFunction::Load(Element* el, FGPropertyValue* var, FGFDMExec* fdmex,
310 const string& Prefix)
312 Name = el->GetAttributeValue(
"name");
313 Element* element = el->GetElement();
315 auto sum = [](
const decltype(Parameters)& Parameters)->double {
318 for (
auto p: Parameters)
325 string operation = element->GetName();
328 if (operation ==
"property" || operation ==
"p") {
329 string property_name = element->GetDataLine();
331 if (var && simgear::strutils::strip(property_name) ==
"#")
332 Parameters.push_back(var);
334 if (property_name.find(
"#") != string::npos) {
335 if (is_number(Prefix)) {
336 property_name = replace(property_name,
"#",Prefix);
339 cerr << element->ReadFrom()
340 <<
fgred <<
"Illegal use of the special character '#'"
342 throw BaseException(
"Fatal Error.");
346 if (element->HasAttribute(
"apply")) {
347 string function_str = element->GetAttributeValue(
"apply");
348 auto f = fdmex->GetTemplateFunc(function_str);
350 Parameters.push_back(
new FGFunctionValue(property_name,
351 PropertyManager, f, element));
353 cerr << element->ReadFrom()
355 << function_str <<
" has been defined. This property will "
356 <<
"not be logged. You should check your configuration file."
361 Parameters.push_back(
new FGPropertyValue(property_name,
362 PropertyManager, element));
364 }
else if (operation ==
"value" || operation ==
"v") {
365 Parameters.push_back(
new FGRealValue(element->GetDataAsNumber()));
366 }
else if (operation ==
"pi") {
367 Parameters.push_back(
new FGRealValue(M_PI));
368 }
else if (operation ==
"table" || operation ==
"t") {
369 string call_type = element->GetAttributeValue(
"type");
370 if (call_type ==
"internal") {
371 std::cerr << el->ReadFrom()
372 <<
"An internal table cannot be nested within a function."
374 throw BaseException(
"An internal table cannot be nested within a function.");
376 Parameters.push_back(
new FGTable(PropertyManager, element, Prefix));
378 }
else if (operation ==
"product") {
379 auto f = [](
const decltype(Parameters)& Parameters)->double {
382 for (
auto p: Parameters)
387 Parameters.push_back(VarArgsFn<
decltype(f)>(f, fdmex, element, Prefix, var));
388 }
else if (operation ==
"sum") {
389 Parameters.push_back(VarArgsFn<
decltype(sum)>(sum, fdmex, element, Prefix, var));
390 }
else if (operation ==
"avg") {
391 auto avg = [&](
const decltype(Parameters)& p)->double {
392 return sum(p) / p.size();
394 Parameters.push_back(VarArgsFn<
decltype(avg)>(avg, fdmex, element, Prefix, var));
395 }
else if (operation ==
"difference") {
396 auto f = [](
const decltype(Parameters)& Parameters)->double {
397 double temp = Parameters[0]->GetValue();
399 for (
auto p = Parameters.begin()+1; p != Parameters.end(); ++p)
400 temp -= (*p)->GetValue();
404 Parameters.push_back(VarArgsFn<
decltype(f)>(f, fdmex, element, Prefix, var));
405 }
else if (operation ==
"min") {
406 auto f = [](
const decltype(Parameters)& Parameters)->double {
407 double _min = HUGE_VAL;
409 for (
auto p : Parameters) {
410 double x = p->GetValue();
417 Parameters.push_back(VarArgsFn<
decltype(f)>(f, fdmex, element, Prefix, var));
418 }
else if (operation ==
"max") {
419 auto f = [](
const decltype(Parameters)& Parameters)->double {
420 double _max = -HUGE_VAL;
422 for (
auto p : Parameters) {
423 double x = p->GetValue();
430 Parameters.push_back(VarArgsFn<
decltype(f)>(f, fdmex, element, Prefix, var));
431 }
else if (operation ==
"and") {
432 string ctxMsg = element->ReadFrom();
433 auto f = [ctxMsg](
const decltype(Parameters)& Parameters)->double {
434 for (
auto p : Parameters) {
435 if (!GetBinary(p->GetValue(), ctxMsg))
441 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix,
443 }
else if (operation ==
"or") {
444 string ctxMsg = element->ReadFrom();
445 auto f = [ctxMsg](
const decltype(Parameters)& Parameters)->double {
446 for (
auto p : Parameters) {
447 if (GetBinary(p->GetValue(), ctxMsg))
453 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix,
455 }
else if (operation ==
"quotient") {
456 auto f = [](
const decltype(Parameters)& p)->double {
457 double y = p[1]->GetValue();
458 return y != 0.0 ? p[0]->GetValue()/y : HUGE_VAL;
460 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
461 }
else if (operation ==
"pow") {
462 auto f = [](
const decltype(Parameters)& p)->double {
465 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
466 }
else if (operation ==
"toradians") {
467 auto f = [](
const decltype(Parameters)& p)->double {
468 return p[0]->GetValue()*M_PI/180.;
470 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
471 }
else if (operation ==
"todegrees") {
472 auto f = [](
const decltype(Parameters)& p)->double {
473 return p[0]->GetValue()*180./M_PI;
475 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
476 }
else if (operation ==
"sqrt") {
477 auto f = [](
const decltype(Parameters)& p)->double {
478 double x = p[0]->GetValue();
479 return x >= 0.0 ? sqrt(x) : -HUGE_VAL;
481 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
482 }
else if (operation ==
"log2") {
483 auto f = [](
const decltype(Parameters)& p)->double {
484 double x = p[0]->GetValue();
485 return x > 0.0 ? log10(x)*invlog2val : -HUGE_VAL;
487 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
488 }
else if (operation ==
"ln") {
489 auto f = [](
const decltype(Parameters)& p)->double {
490 double x = p[0]->GetValue();
491 return x > 0.0 ? log(x) : -HUGE_VAL;
493 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
494 }
else if (operation ==
"log10") {
495 auto f = [](
const decltype(Parameters)& p)->double {
496 double x = p[0]->GetValue();
497 return x > 0.0 ? log10(x) : -HUGE_VAL;
499 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
500 }
else if (operation ==
"sign") {
501 auto f = [](
const decltype(Parameters)& p)->double {
502 return p[0]->GetValue() < 0.0 ? -1 : 1;
504 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
505 }
else if (operation ==
"exp") {
506 Parameters.push_back(make_MathFn(exp, fdmex, element, Prefix, var));
507 }
else if (operation ==
"abs") {
508 Parameters.push_back(make_MathFn(fabs, fdmex, element, Prefix, var));
509 }
else if (operation ==
"sin") {
510 Parameters.push_back(make_MathFn(sin, fdmex, element, Prefix, var));
511 }
else if (operation ==
"cos") {
512 Parameters.push_back(make_MathFn(cos, fdmex, element, Prefix, var));
513 }
else if (operation ==
"tan") {
514 Parameters.push_back(make_MathFn(tan, fdmex, element, Prefix, var));
515 }
else if (operation ==
"asin") {
516 Parameters.push_back(make_MathFn(asin, fdmex, element, Prefix, var));
517 }
else if (operation ==
"acos") {
518 Parameters.push_back(make_MathFn(acos, fdmex, element, Prefix, var));
519 }
else if (operation ==
"atan") {
520 Parameters.push_back(make_MathFn(atan, fdmex, element, Prefix, var));
521 }
else if (operation ==
"floor") {
522 Parameters.push_back(make_MathFn(floor, fdmex, element, Prefix, var));
523 }
else if (operation ==
"ceil") {
524 Parameters.push_back(make_MathFn(ceil, fdmex, element, Prefix, var));
525 }
else if (operation ==
"fmod") {
526 auto f = [](
const decltype(Parameters)& p)->double {
527 double y = p[1]->GetValue();
528 return y != 0.0 ? fmod(p[0]->
GetValue(), y) : HUGE_VAL;
530 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
531 }
else if (operation ==
"roundmultiple") {
532 if (element->GetNumElements() == 1)
533 Parameters.push_back(make_MathFn(round, fdmex, element, Prefix, var));
535 auto f = [](
const decltype(Parameters)& p)->double {
536 double multiple = p[1]->GetValue();
537 return round((p[0]->
GetValue() / multiple)) * multiple;
539 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var, 2));
541 }
else if (operation ==
"atan2") {
542 auto f = [](
const decltype(Parameters)& p)->double {
545 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
546 }
else if (operation ==
"mod") {
547 auto f = [](
const decltype(Parameters)& p)->double {
548 return static_cast<int>(p[0]->GetValue()) %
static_cast<int>(p[1]->
GetValue());
550 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
551 }
else if (operation ==
"fraction") {
552 auto f = [](
const decltype(Parameters)& p)->double {
554 return modf(p[0]->
GetValue(), &scratch);
556 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
557 }
else if (operation ==
"integer") {
558 auto f = [](
const decltype(Parameters)& p)->double {
563 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
564 }
else if (operation ==
"lt") {
565 auto f = [](
const decltype(Parameters)& p)->double {
566 return p[0]->GetValue() < p[1]->GetValue() ? 1.0 : 0.0;
568 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
569 }
else if (operation ==
"le") {
570 auto f = [](
const decltype(Parameters)& p)->double {
571 return p[0]->GetValue() <= p[1]->GetValue() ? 1.0 : 0.0;
573 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
574 }
else if (operation ==
"gt") {
575 auto f = [](
const decltype(Parameters)& p)->double {
576 return p[0]->GetValue() > p[1]->GetValue() ? 1.0 : 0.0;
578 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
579 }
else if (operation ==
"ge") {
580 auto f = [](
const decltype(Parameters)& p)->double {
581 return p[0]->GetValue() >= p[1]->GetValue() ? 1.0 : 0.0;
583 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
584 }
else if (operation ==
"eq") {
585 auto f = [](
const decltype(Parameters)& p)->double {
586 return p[0]->GetValue() == p[1]->GetValue() ? 1.0 : 0.0;
588 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
589 }
else if (operation ==
"nq") {
590 auto f = [](
const decltype(Parameters)& p)->double {
591 return p[0]->GetValue() != p[1]->GetValue() ? 1.0 : 0.0;
593 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix, var));
594 }
else if (operation ==
"not") {
595 string ctxMsg = element->ReadFrom();
596 auto f = [ctxMsg](
const decltype(Parameters)& p)->double {
597 return GetBinary(p[0]->
GetValue(), ctxMsg) ? 0.0 : 1.0;
599 Parameters.push_back(
new aFunc<
decltype(f), 1>(f, fdmex, element, Prefix, var));
600 }
else if (operation ==
"ifthen") {
601 string ctxMsg = element->ReadFrom();
602 auto f = [ctxMsg](
const decltype(Parameters)& p)->double {
603 if (GetBinary(p[0]->
GetValue(), ctxMsg))
604 return p[1]->GetValue();
606 return p[2]->GetValue();
608 Parameters.push_back(
new aFunc<
decltype(f), 3>(f, fdmex, element, Prefix, var));
609 }
else if (operation ==
"random") {
612 string mean_attr = element->GetAttributeValue(
"mean");
613 string stddev_attr = element->GetAttributeValue(
"stddev");
614 if (!mean_attr.empty()) {
616 mean = atof_locale_c(mean_attr);
617 }
catch (InvalidNumber& e) {
618 cerr << element->ReadFrom() << e.what() << endl;
622 if (!stddev_attr.empty()) {
624 stddev = atof_locale_c(stddev_attr);
625 }
catch (InvalidNumber& e) {
626 cerr << element->ReadFrom() << e.what() << endl;
630 auto generator(makeRandomGenerator(element, fdmex));
631 auto f = [generator, mean, stddev]()->
double {
632 double value = generator->GetNormalRandomNumber();
633 return value*stddev + mean;
635 Parameters.push_back(
new aFunc<
decltype(f), 0>(f, PropertyManager, element,
637 }
else if (operation ==
"urandom") {
640 string lower_attr = element->GetAttributeValue(
"lower");
641 string upper_attr = element->GetAttributeValue(
"upper");
642 if (!lower_attr.empty()) {
644 lower = atof_locale_c(lower_attr);
645 }
catch (InvalidNumber &e) {
646 cerr << element->ReadFrom() << e.what() << endl;
650 if (!upper_attr.empty()) {
652 upper = atof_locale_c(upper_attr);
653 }
catch (InvalidNumber &e) {
654 cerr << element->ReadFrom() << e.what() << endl;
658 auto generator(makeRandomGenerator(element, fdmex));
659 double a = 0.5*(upper-lower);
660 double b = 0.5*(upper+lower);
661 auto f = [generator, a, b]()->
double {
662 double value = generator->GetUniformRandomNumber();
665 Parameters.push_back(
new aFunc<
decltype(f), 0>(f, PropertyManager, element,
667 }
else if (operation ==
"switch") {
668 string ctxMsg = element->ReadFrom();
669 auto f = [ctxMsg](
const decltype(Parameters)& p)->double {
670 double temp = p[0]->GetValue();
673 <<
"The switch function index (" << temp
674 <<
") is negative." <<
reset << endl;
675 throw BaseException(
"Fatal error");
677 size_t n = p.size()-1;
678 size_t i =
static_cast<size_t>(temp+0.5);
681 return p[i+1]->GetValue();
684 <<
"The switch function index (" << temp
685 <<
") selected a value above the range of supplied values"
686 <<
"[0:" << n-1 <<
"]"
687 <<
" - not enough values were supplied." <<
reset << endl;
688 throw BaseException(
"Fatal error");
691 Parameters.push_back(
new aFunc<
decltype(f), 2>(f, fdmex, element, Prefix,
693 }
else if (operation ==
"interpolate1d") {
694 auto f = [](
const decltype(Parameters)& p)->double {
698 double x = p[0]->GetValue();
699 double xmin = p[1]->GetValue();
700 double ymin = p[2]->GetValue();
701 if (x <= xmin)
return ymin;
703 double xmax = p[n-2]->GetValue();
704 double ymax = p[n-1]->GetValue();
705 if (x >= xmax)
return ymax;
708 size_t nmax = (n-3)/2;
709 while (nmax-nmin > 1) {
710 size_t m = (nmax-nmin)/2+nmin;
711 double xm = p[2*m+1]->GetValue();
712 double ym = p[2*m+2]->GetValue();
726 return ymin + (x-xmin)*(ymax-ymin)/(xmax-xmin);
728 Parameters.push_back(
new aFunc<
decltype(f), 5>(f, fdmex, element, Prefix,
729 var, MaxArgs, OddEven::Odd));
730 }
else if (operation ==
"rotation_alpha_local") {
734 auto f = [](
const decltype(Parameters)& p)->double {
735 double alpha = p[0]->GetValue()*degtorad;
736 double beta = p[1]->GetValue()*degtorad;
737 double phi = p[3]->GetValue()*degtorad;
738 double theta = p[4]->GetValue()*degtorad;
739 double psi = p[5]->GetValue()*degtorad;
741 FGQuaternion qTb2l(phi, theta, psi);
742 double cos_beta = cos(beta);
743 FGColumnVector3 wind_body(cos(alpha)*cos_beta, sin(beta),
744 sin(alpha)*cos_beta);
745 FGColumnVector3 wind_local = qTb2l.GetT()*wind_body;
747 if (fabs(fabs(wind_local(eY)) - 1.0) < 1E-9)
750 return atan2(wind_local(eZ), wind_local(eX))*radtodeg;
752 Parameters.push_back(
new aFunc<
decltype(f), 6>(f, fdmex, element, Prefix, var));
753 }
else if (operation ==
"rotation_beta_local") {
757 auto f = [](
const decltype(Parameters)& p)->double {
758 double alpha = p[0]->GetValue()*degtorad;
759 double beta = p[1]->GetValue()*degtorad;
760 double phi = p[3]->GetValue()*degtorad;
761 double theta = p[4]->GetValue()*degtorad;
762 double psi = p[5]->GetValue()*degtorad;
763 FGQuaternion qTb2l(phi, theta, psi);
764 double cos_beta = cos(beta);
765 FGColumnVector3 wind_body(cos(alpha)*cos_beta, sin(beta),
766 sin(alpha)*cos_beta);
767 FGColumnVector3 wind_local = qTb2l.GetT()*wind_body;
769 if (fabs(fabs(wind_local(eY)) - 1.0) < 1E-9)
770 return wind_local(eY) > 0.0 ? 0.5*M_PI : -0.5*M_PI;
772 double alpha_local = atan2(wind_local(eZ), wind_local(eX));
773 double cosa = cos(alpha_local);
774 double sina = sin(alpha_local);
777 if (fabs(cosa) > fabs(sina))
778 cosb = wind_local(eX) / cosa;
780 cosb = wind_local(eZ) / sina;
782 return atan2(wind_local(eY), cosb)*radtodeg;
784 Parameters.push_back(
new aFunc<
decltype(f), 6>(f, fdmex, element, Prefix, var));
785 }
else if (operation ==
"rotation_gamma_local") {
789 auto f = [](
const decltype(Parameters)& p)->double {
790 double alpha = p[0]->GetValue()*degtorad;
791 double beta = p[1]->GetValue()*degtorad;
792 double gamma = p[2]->GetValue()*degtorad;
793 double phi = p[3]->GetValue()*degtorad;
794 double theta = p[4]->GetValue()*degtorad;
795 double psi = p[5]->GetValue()*degtorad;
796 double cos_alpha = cos(alpha), sin_alpha = sin(alpha);
797 double cos_beta = cos(beta), sin_beta = sin(beta);
798 double cos_gamma = cos(gamma), sin_gamma = sin(gamma);
799 FGQuaternion qTb2l(phi, theta, psi);
800 FGColumnVector3 wind_body_X(cos_alpha*cos_beta, sin_beta,
802 FGColumnVector3 wind_body_Y(-sin_alpha*sin_gamma-sin_beta*cos_alpha*cos_gamma,
804 -sin_alpha*sin_beta*cos_gamma+sin_gamma*cos_alpha);
805 FGColumnVector3 wind_local_X = qTb2l.GetT()*wind_body_X;
806 FGColumnVector3 wind_local_Y = qTb2l.GetT()*wind_body_Y;
807 double cosacosb = wind_local_X(eX);
808 double sinb = wind_local_X(eY);
809 double sinacosb = wind_local_X(eZ);
812 if (fabs(sinb) < 1E-9) {
813 cosc = wind_local_Y(eY);
815 if (fabs(cosacosb) > fabs(sinacosb))
816 sinc = wind_local_Y(eZ) / cosacosb;
818 sinc = -wind_local_Y(eX) / sinacosb;
820 else if (fabs(fabs(sinb)-1.0) < 1E-9) {
821 sinc = wind_local_Y(eZ);
822 cosc = -wind_local_Y(eX);
825 sinc = cosacosb*wind_local_Y(eZ)-sinacosb*wind_local_Y(eX);
826 cosc = (-sinacosb*wind_local_Y(eZ)-cosacosb*wind_local_Y(eX))/sinb;
829 return atan2(sinc, cosc)*radtodeg;
831 Parameters.push_back(
new aFunc<
decltype(f), 6>(f, fdmex, element, Prefix, var));
832 }
else if (operation ==
"rotation_bf_to_wf") {
835 string ctxMsg = element->ReadFrom();
836 auto f = [ctxMsg](
const decltype(Parameters)& p)->double {
837 double rx = p[0]->GetValue();
838 double ry = p[1]->GetValue();
839 double rz = p[2]->GetValue();
840 double alpha = p[3]->GetValue()*degtorad;
841 double beta = p[4]->GetValue()*degtorad;
842 double gamma = p[5]->GetValue()*degtorad;
843 int idx =
static_cast<int>(p[6]->GetValue());
845 if ((idx < 1) || (idx > 3)) {
847 <<
"The index must be one of the integer value 1, 2 or 3."
849 throw BaseException(
"Fatal error");
852 FGQuaternion qa(eY, -alpha), qb(eZ, beta), qc(eX, -gamma);
853 FGMatrix33 mT = (qa*qb*qc).GetT();
854 FGColumnVector3 r0(rx, ry, rz);
855 FGColumnVector3 r = mT*r0;
859 Parameters.push_back(
new aFunc<
decltype(f), 7>(f, fdmex, element, Prefix, var));
860 }
else if (operation ==
"rotation_wf_to_bf") {
863 string ctxMsg = element->ReadFrom();
864 auto f = [ctxMsg](
const decltype(Parameters)& p)->double {
865 double rx = p[0]->GetValue();
866 double ry = p[1]->GetValue();
867 double rz = p[2]->GetValue();
868 double alpha = p[3]->GetValue()*degtorad;
869 double beta = p[4]->GetValue()*degtorad;
870 double gamma = p[5]->GetValue()*degtorad;
871 int idx =
static_cast<int>(p[6]->GetValue());
873 if ((idx < 1) || (idx > 3)) {
875 <<
"The index must be one of the integer value 1, 2 or 3."
877 throw BaseException(
"Fatal error");
880 FGQuaternion qa(eY, -alpha), qb(eZ, beta), qc(eX, -gamma);
881 FGMatrix33 mT = (qa*qb*qc).GetT();
882 FGColumnVector3 r0(rx, ry, rz);
884 FGColumnVector3 r = mT*r0;
888 Parameters.push_back(
new aFunc<
decltype(f), 7>(f, fdmex, element, Prefix, var));
889 }
else if (operation !=
"description") {
891 <<
"Bad operation <" << operation
892 <<
"> detected in configuration file" <<
reset << endl;
897 if (!Parameters.empty()){
900 if (p && p->IsConstant()) {
901 double constant = p->GetValue();
902 FGPropertyNode_ptr node = p->pNode;
903 string pName = p->GetName();
905 Parameters.pop_back();
906 Parameters.push_back(
new FGRealValue(constant));
909 <<
"<" << operation <<
"> is applied on constant parameters."
910 << endl <<
"It will be replaced by its result ("
914 node->setDoubleValue(constant);
915 node->setAttribute(SGPropertyNode::WRITE,
false);
917 cout <<
" and the property " << pName
918 <<
" will be unbound and made read only.";
920 cout <<
reset << endl << endl;
923 element = el->GetNextElement();
935 if (pNode && pNode->isTied())
936 PropertyManager->Untie(pNode);
945 for (
auto p: Parameters) {
946 if (!p->IsConstant())
969 if (cached)
return cachedValue;
971 double val = Parameters[0]->GetValue();
973 if (pCopyTo) pCopyTo->setDoubleValue(val);
982 ostringstream buffer;
984 buffer << setw(9) << setprecision(6) <<
GetValue();
990string FGFunction::CreateOutputNode(
Element* el,
const string& Prefix)
994 if ( !Name.empty() ) {
996 nName = PropertyManager->mkPropertyName(Name,
false);
998 if (is_number(Prefix)) {
999 if (Name.find(
"#") != string::npos) {
1000 Name = replace(Name,
"#",Prefix);
1001 nName = PropertyManager->mkPropertyName(Name,
false);
1004 <<
"Malformed function name with number: " << Prefix
1005 <<
" and property name: " << Name
1006 <<
" but no \"#\" sign for substitution." << endl;
1009 nName = PropertyManager->mkPropertyName(Prefix +
"/" + Name,
false);
1013 pNode = PropertyManager->GetNode(nName,
true);
1014 if (pNode->isTied()) {
1016 <<
"Property " << nName <<
" has already been successfully bound (late)." << endl;
1017 throw BaseException(
"Failed to bind the property to an existing already tied node.");
1026void FGFunction::bind(Element* el,
const string& Prefix)
1028 string nName = CreateOutputNode(el, Prefix);
1053void FGFunction::Debug(
int from)
1055 if (debug_lvl <= 0)
return;
1057 if (debug_lvl & 1) {
1060 cout <<
" Function: " << Name << endl;
1063 if (debug_lvl & 2 ) {
1064 if (from == 0) cout <<
"Instantiated: FGFunction" << endl;
1065 if (from == 1) cout <<
"Destroyed: FGFunction" << endl;
1067 if (debug_lvl & 4 ) {
1069 if (debug_lvl & 8 ) {
1071 if (debug_lvl & 16) {
1073 if (debug_lvl & 64) {
const std::string & GetName(void) const
Retrieves the element name.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
unsigned int GetNumElements(void)
Returns the number of child elements for this element.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
Encapsulates the JSBSim simulation executive.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Represents a mathematical function.
FGFunction()
Default constructor.
double GetValue(void) const override
Retrieves the value of the function object.
bool IsConstant(void) const override
Does the function always return the same result (i.e.
void cacheValue(bool shouldCache)
Specifies whether to cache the value of the function, so it is calculated only once per frame.
std::string GetValueAsString(void) const
The value that the function evaluates to, as a string.
~FGFunction(void) override
Destructor Make sure the function is untied before destruction.
static char fggreen[6]
green text
static char fgred[6]
red text
static char reset[5]
resets text properties
static char highint[5]
highlights text
Represents various types of parameters.
Represents a property value which can use late binding.