39 G4bool DoCheck(
const G4UIparameter& p,
const char* newValue);
40 G4bool DoCheck(
const G4UIcommand& cmd,
const char* newValue);
43 G4bool RangeCheckImpl(
const char* newValue);
46 unsigned IndexOf(
const char* nam);
49 unsigned IsParameter(
const char* nam);
52 using yystype = G4UItokenNum::yystype;
57 yystype LogicalORExpression();
58 yystype LogicalANDExpression();
59 yystype EqualityExpression();
60 yystype RelationalExpression();
61 yystype AdditiveExpression();
62 yystype MultiplicativeExpression();
63 yystype UnaryExpression();
64 yystype PrimaryExpression();
66 G4int Eval2(
const yystype& arg1,
G4int op,
const yystype& arg2);
75 G4String rangeExpression;
77 std::vector<const G4UIparameter*> parameter;
83 std::vector<yystype> newVal;
100 return RangeCheckImpl(newValue);
103inline G4bool G4UIRangeChecker::DoCheck(
const G4UIcommand& cmd,
const char* newValue)
110 newVal.resize(parameter.size());
111 for (
G4int i = 0; i < (
G4int)parameter.size(); ++i) {
115 return RangeCheckImpl(newValue);
118inline G4bool G4UIRangeChecker::RangeCheckImpl(
const char* newValue)
120 if (rangeExpression.empty()) {
126 std::istringstream is(newValue);
127 for (
unsigned i = 0; i < parameter.size(); ++i) {
128 char type = (char)std::toupper(parameter[i]->GetParameterType());
149 result = Expression();
155 G4cerr <<
"Illegal Expression in parameter range." <<
G4endl;
161 G4cerr <<
"parameter out of range: " << rangeExpression <<
G4endl;
165inline unsigned G4UIRangeChecker::IndexOf(
const char* nam)
167 for (
unsigned i = 0; i < parameter.size(); ++i) {
168 if (parameter[i]->GetParameterName() == nam) {
173 G4cerr <<
"parameter name:" << nam <<
" not found." <<
G4endl;
177inline unsigned G4UIRangeChecker::IsParameter(
const char* nam)
179 for (
auto& i : parameter) {
180 if (i->GetParameterName() == nam) {
189inline yystype G4UIRangeChecker::Expression()
191 return LogicalORExpression();
195inline yystype G4UIRangeChecker::LogicalORExpression()
198 yystype p = LogicalANDExpression();
203 G4cerr <<
"Parameter range: illegal type at '||'" <<
G4endl;
209 p = LogicalANDExpression();
211 G4cerr <<
"Parameter range: illegal type at '||'" <<
G4endl;
220 result.
I +=
static_cast<int>(p.
L != 0L);
224 result.
I +=
static_cast<int>(p.
D != 0.0);
236inline yystype G4UIRangeChecker::LogicalANDExpression()
239 yystype p = EqualityExpression();
244 G4cerr <<
"Parameter range: illegal type at '&&'" <<
G4endl;
250 p = EqualityExpression();
252 G4cerr <<
"Parameter range: illegal type at '&&'" <<
G4endl;
261 result.
I *=
static_cast<int>(p.
L != 0L);
265 result.
I *=
static_cast<int>(p.
D != 0.0);
277inline yystype G4UIRangeChecker::EqualityExpression()
281 yystype result = RelationalExpression();
282 if (token ==
EQ || token ==
NE) {
286 arg2 = RelationalExpression();
287 result.
I = Eval2(arg1, operat, arg2);
292 G4cerr <<
"Parameter range: error at EqualityExpression" <<
G4endl;
300inline yystype G4UIRangeChecker::RelationalExpression()
306 arg1 = AdditiveExpression();
307 if (token ==
GT || token ==
GE || token ==
LT || token ==
LE) {
310 arg2 = AdditiveExpression();
311 result.
I = Eval2(arg1, operat, arg2);
315 result = std::move(arg1);
321inline yystype G4UIRangeChecker::AdditiveExpression()
323 yystype result = MultiplicativeExpression();
324 if (token !=
'+' && token !=
'-') {
327 G4cerr <<
"Parameter range: operator " << (char)token <<
" is not supported." <<
G4endl;
333inline yystype G4UIRangeChecker::MultiplicativeExpression()
335 yystype result = UnaryExpression();
336 if (token !=
'*' && token !=
'/' && token !=
'%') {
339 G4cerr <<
"Parameter range: operator " << (char)token <<
" is not supported." <<
G4endl;
345inline yystype G4UIRangeChecker::UnaryExpression()
352 p = UnaryExpression();
368 result = UnaryExpression();
372 G4cerr <<
"Parameter range error: "
373 <<
"operator '!' is not supported (sorry)." <<
G4endl;
375 result = UnaryExpression();
378 result = PrimaryExpression();
384inline yystype G4UIRangeChecker::PrimaryExpression()
410 result = Expression();
425inline G4int G4UIRangeChecker::Eval2(
const yystype& arg1,
G4int op,
const yystype& arg2)
428 G4cerr << commandName <<
": meaningless comparison" <<
G4int(arg1.
type) <<
" "
437 unsigned i = IndexOf(arg1.
S);
438 newValtype = (char)std::toupper(parameter[i]->GetParameterType());
439 switch (newValtype) {
447 unsigned iii = IndexOf(arg2.
S);
448 char newValtype2 = (char)std::toupper(parameter[iii]->GetParameterType());
449 if (newValtype2 ==
'I') {
452 if (newValtype2 ==
'L') {
453 G4cerr <<
"Warning : Integer is compared with long int : " << rangeExpression <<
G4endl;
456 if (newValtype2 ==
'D') {
457 G4cerr <<
"Warning : Integer is compared with double : " << rangeExpression <<
G4endl;
463 G4cerr <<
"integer operand expected for " << rangeExpression <<
'.' <<
G4endl;
474 unsigned iii = IndexOf(arg2.
S);
475 char newValtype2 = (char)std::toupper(parameter[iii]->GetParameterType());
476 if (newValtype2 ==
'I') {
479 if (newValtype2 ==
'L') {
482 if (newValtype2 ==
'D') {
483 G4cerr <<
"Warning : Long int is compared with double : " << rangeExpression <<
G4endl;
489 G4cerr <<
"integer operand expected for " << rangeExpression <<
'.' <<
G4endl;
505 unsigned iii = IndexOf(arg2.
S);
506 char newValtype2 = (char)std::toupper(parameter[iii]->GetParameterType());
507 if (newValtype2 ==
'I') {
510 if (newValtype2 ==
'L') {
513 if (newValtype2 ==
'D') {
523 unsigned i = IndexOf(arg2.
S);
524 newValtype = (char)std::toupper(parameter[i]->GetParameterType());
525 switch (newValtype) {
531 G4cerr <<
"integer operand expected for " << rangeExpression <<
'.' <<
G4endl;
539 G4cerr <<
"long int operand expected for " << rangeExpression <<
'.' <<
G4endl;
558inline tokenNum G4UIRangeChecker::Yylex()
563 while ((c = G4UIpGetc()) ==
' ' || c ==
'\t' || c ==
'\n') {
570 if ((isdigit(c) != 0) || c ==
'.') {
572 buf += (
unsigned char)c;
574 }
while (c ==
'.' || (isdigit(c) != 0) || c ==
'e' || c ==
'E' || c ==
'+' || c ==
'-');
577 std::istringstream is(t);
590 if ((isalpha(c) != 0) || c ==
'_') {
592 buf += (
unsigned char)c;
593 }
while ((c = G4UIpGetc()) != EOF && ((isalnum(c) != 0) || c ==
'_'));
595 if (IsParameter(buf) != 0u) {
596 yylval.
S = std::move(buf);
624 G4int c = G4UIpGetc();
634inline G4int G4UIRangeChecker::G4UIpGetc()
636 std::size_t length = rangeExpression.length();
637 if (bp < (
G4int)length) {
638 return rangeExpression[bp++];
645inline G4int G4UIRangeChecker::G4UIpUngetc(
G4int c)
650 if (bp > 0 && c == rangeExpression[bp - 1]) {
666 return r.DoCheck(p, value);
672 return r.DoCheck(p, value);
G4double D(G4double temp)
G4GLOB_DLL std::ostream G4cerr
std::size_t GetParameterEntries() const
G4UIparameter * GetParameter(G4int i) const
const G4String & GetCommandName() const
const G4String & GetRange() const
const G4String & GetParameterRange() const
const G4String & GetParameterName() const
G4bool RangeCheck(const G4UIparameter &p, const char *value)
G4bool IsDouble(const char *str)
G4int CompareLong(G4long arg1, G4int op, G4long arg2, G4int &errCode)
G4int CompareDouble(G4double arg1, G4int op, G4double arg2, G4int &errCode)
G4bool IsInt(const char *str, short maxDigits)
G4int CompareInt(G4int arg1, G4int op, G4int arg2, G4int &errCode)