1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2008 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 * 25 * check.c -- routines for checking the prop tree 26 * 27 * this module provides semantic checks on the parse tree. most of 28 * these checks happen during the construction of the parse tree, 29 * when the various tree_X() routines call the various check_X() 30 * routines. in a couple of special cases, a check function will 31 * process the parse tree after it has been fully constructed. these 32 * cases are noted in the comments above the check function. 33 */ 34 35 #pragma ident "%Z%%M% %I% %E% SMI" 36 37 #include <stdio.h> 38 #include "out.h" 39 #include "stable.h" 40 #include "literals.h" 41 #include "lut.h" 42 #include "tree.h" 43 #include "ptree.h" 44 #include "check.h" 45 46 static int check_reportlist(enum nodetype t, const char *s, struct node *np); 47 static int check_num(enum nodetype t, const char *s, struct node *np); 48 static int check_quote(enum nodetype t, const char *s, struct node *np); 49 static int check_action(enum nodetype t, const char *s, struct node *np); 50 static int check_num_func(enum nodetype t, const char *s, struct node *np); 51 static int check_fru_asru(enum nodetype t, const char *s, struct node *np); 52 static int check_engine(enum nodetype t, const char *s, struct node *np); 53 static int check_count(enum nodetype t, const char *s, struct node *np); 54 static int check_timeval(enum nodetype t, const char *s, struct node *np); 55 static int check_id(enum nodetype t, const char *s, struct node *np); 56 static int check_serd_method(enum nodetype t, const char *s, struct node *np); 57 static int check_serd_id(enum nodetype t, const char *s, struct node *np); 58 static int check_nork(struct node *np); 59 static void check_cycle_lhs(struct node *stmtnp, struct node *arrow); 60 static void check_cycle_lhs_try(struct node *stmtnp, struct node *lhs, 61 struct node *rhs); 62 static void check_cycle_rhs(struct node *rhs); 63 static void check_proplists_lhs(enum nodetype t, struct node *lhs); 64 65 static struct { 66 enum nodetype t; 67 const char *name; 68 int required; 69 int (*checker)(enum nodetype t, const char *s, struct node *np); 70 int outflags; 71 } Allowednames[] = { 72 { T_FAULT, "FITrate", 0, check_num_func, O_ERR }, 73 { T_FAULT, "FRU", 0, check_fru_asru, O_ERR }, 74 { T_FAULT, "ASRU", 0, check_fru_asru, O_ERR }, 75 { T_FAULT, "message", 0, check_num_func, O_ERR }, 76 { T_FAULT, "retire", 0, check_num_func, O_ERR }, 77 { T_FAULT, "response", 0, check_num_func, O_ERR }, 78 { T_FAULT, "action", 0, check_action, O_ERR }, 79 { T_FAULT, "count", 0, check_count, O_ERR }, 80 { T_FAULT, "engine", 0, check_engine, O_ERR }, 81 { T_UPSET, "engine", 0, check_engine, O_ERR }, 82 { T_DEFECT, "FRU", 0, check_fru_asru, O_ERR }, 83 { T_DEFECT, "ASRU", 0, check_fru_asru, O_ERR }, 84 { T_DEFECT, "engine", 0, check_engine, O_ERR }, 85 { T_EREPORT, "poller", 0, check_id, O_ERR }, 86 { T_EREPORT, "delivery", 0, check_timeval, O_ERR }, 87 { T_EREPORT, "discard_if_config_unknown", 0, check_num, O_ERR }, 88 { T_SERD, "N", 1, check_num, O_ERR }, 89 { T_SERD, "T", 1, check_timeval, O_ERR }, 90 { T_SERD, "method", 0, check_serd_method, O_ERR }, 91 { T_SERD, "trip", 0, check_reportlist, O_ERR }, 92 { T_SERD, "FRU", 0, check_fru_asru, O_ERR }, 93 { T_SERD, "id", 0, check_serd_id, O_ERR }, 94 { T_ERROR, "ASRU", 0, check_fru_asru, O_ERR }, 95 { T_CONFIG, NULL, 0, check_quote, O_ERR }, 96 { 0, NULL, 0 }, 97 }; 98 99 void 100 check_init(void) 101 { 102 int i; 103 104 for (i = 0; Allowednames[i].t; i++) 105 if (Allowednames[i].name != NULL) 106 Allowednames[i].name = stable(Allowednames[i].name); 107 } 108 109 void 110 check_fini(void) 111 { 112 } 113 114 /*ARGSUSED*/ 115 void 116 check_report_combination(struct node *np) 117 { 118 /* nothing to check for here. poller is only prop and it is optional */ 119 } 120 121 /* 122 * check_path_iterators -- verify all iterators are explicit 123 */ 124 static void 125 check_path_iterators(struct node *np) 126 { 127 if (np == NULL) 128 return; 129 130 switch (np->t) { 131 case T_ARROW: 132 check_path_iterators(np->u.arrow.lhs); 133 check_path_iterators(np->u.arrow.rhs); 134 break; 135 136 case T_LIST: 137 check_path_iterators(np->u.expr.left); 138 check_path_iterators(np->u.expr.right); 139 break; 140 141 case T_EVENT: 142 check_path_iterators(np->u.event.epname); 143 break; 144 145 case T_NAME: 146 if (np->u.name.child == NULL) 147 outfl(O_DIE, np->file, np->line, 148 "internal error: check_path_iterators: " 149 "unexpected implicit iterator: %s", 150 np->u.name.s); 151 check_path_iterators(np->u.name.next); 152 break; 153 154 default: 155 outfl(O_DIE, np->file, np->line, 156 "internal error: check_path_iterators: " 157 "unexpected type: %s", 158 ptree_nodetype2str(np->t)); 159 } 160 } 161 162 void 163 check_arrow(struct node *np) 164 { 165 ASSERTinfo(np->t == T_ARROW, ptree_nodetype2str(np->t)); 166 167 if (np->u.arrow.lhs->t != T_ARROW && 168 np->u.arrow.lhs->t != T_LIST && 169 np->u.arrow.lhs->t != T_EVENT) { 170 outfl(O_ERR, 171 np->u.arrow.lhs->file, np->u.arrow.lhs->line, 172 "%s not allowed on left-hand side of arrow", 173 ptree_nodetype2str(np->u.arrow.lhs->t)); 174 } 175 176 if (!check_nork(np->u.arrow.nnp) || 177 !check_nork(np->u.arrow.knp)) 178 outfl(O_ERR, np->file, np->line, 179 "counts associated with propagation arrows " 180 "must be integers"); 181 182 check_path_iterators(np); 183 } 184 185 /* 186 * make sure the nork values are valid. 187 * Nork values must be "A" for all(T_NAME), 188 * a number(T_NUM), or a simple 189 * expression(T_SUB, T_ADD, T_MUL, T_DIV) 190 */ 191 static int 192 check_nork(struct node *np) 193 { 194 int rval = 0; 195 196 /* NULL means no nork value which is allowed */ 197 if (np == NULL) { 198 rval = 1; 199 } 200 else 201 { 202 /* if the nork is a name it must be A for "All" */ 203 if (np->t == T_NAME) 204 if (*np->u.name.s == 'A') 205 return (1); 206 207 /* T_NUM allowed */ 208 if (np->t == T_NUM) 209 rval = 1; 210 211 /* simple expressions allowed */ 212 if (np->t == T_SUB || 213 np->t == T_ADD || 214 np->t == T_MUL || 215 np->t == T_DIV) 216 rval = 1; 217 } 218 219 return (rval); 220 } 221 222 static int 223 check_reportlist(enum nodetype t, const char *s, struct node *np) 224 { 225 if (np == NULL) 226 return (1); 227 else if (np->t == T_EVENT) { 228 if (np->u.event.ename->u.name.t != N_EREPORT) { 229 outfl(O_ERR, np->file, np->line, 230 "%s %s property must begin with \"ereport.\"", 231 ptree_nodetype2str(t), s); 232 } else if (tree_event2np_lut_lookup(Ereports, np) == NULL) { 233 outfl(O_ERR, np->file, np->line, 234 "%s %s property contains undeclared name", 235 ptree_nodetype2str(t), s); 236 } 237 check_type_iterator(np); 238 } else if (np->t == T_LIST) { 239 (void) check_reportlist(t, s, np->u.expr.left); 240 (void) check_reportlist(t, s, np->u.expr.right); 241 } 242 return (1); 243 } 244 245 static int 246 check_num(enum nodetype t, const char *s, struct node *np) 247 { 248 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 249 if (np->t != T_NUM) 250 outfl(O_ERR, np->file, np->line, 251 "%s %s property must be a single number", 252 ptree_nodetype2str(t), s); 253 return (1); 254 } 255 256 /*ARGSUSED1*/ 257 static int 258 check_quote(enum nodetype t, const char *s, struct node *np) 259 { 260 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 261 if (np->t != T_QUOTE) 262 outfl(O_ERR, np->file, np->line, 263 "%s properties must be quoted strings", 264 ptree_nodetype2str(t)); 265 return (1); 266 } 267 268 static int 269 check_action(enum nodetype t, const char *s, struct node *np) 270 { 271 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 272 273 if (np->t != T_FUNC) 274 outfl(O_ERR, np->file, np->line, 275 "%s %s property must be a function or list of functions", 276 ptree_nodetype2str(t), s); 277 return (1); 278 } 279 280 static int 281 check_num_func(enum nodetype t, const char *s, struct node *np) 282 { 283 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 284 if (np->t != T_NUM && np->t != T_FUNC) 285 outfl(O_ERR, np->file, np->line, 286 "%s %s property must be a number or function", 287 ptree_nodetype2str(t), s); 288 return (1); 289 } 290 291 static int 292 check_fru_asru(enum nodetype t, const char *s, struct node *np) 293 { 294 ASSERT(s != NULL); 295 296 /* make sure it is a node type T_NAME? */ 297 if (np->t == T_NAME) { 298 if (s == L_ASRU) { 299 if (tree_name2np_lut_lookup_name(ASRUs, np) == NULL) 300 outfl(O_ERR, np->file, np->line, 301 "ASRU property contains undeclared asru"); 302 } else if (s == L_FRU) { 303 if (tree_name2np_lut_lookup_name(FRUs, np) == NULL) 304 outfl(O_ERR, np->file, np->line, 305 "FRU property contains undeclared fru"); 306 } else { 307 outfl(O_ERR, np->file, np->line, 308 "illegal property name in %s declaration: %s", 309 ptree_nodetype2str(t), s); 310 } 311 check_type_iterator(np); 312 } else 313 outfl(O_ERR, np->file, np->line, 314 "illegal type used for %s property: %s", 315 s, ptree_nodetype2str(np->t)); 316 return (1); 317 } 318 319 static int 320 check_engine(enum nodetype t, const char *s, struct node *np) 321 { 322 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 323 if (np->t != T_EVENT) 324 outfl(O_ERR, np->file, np->line, 325 "%s %s property must be an engine name " 326 "(i.e. serd.x or serd.x@a/b)", 327 ptree_nodetype2str(t), s); 328 329 return (1); 330 } 331 332 static int 333 check_count(enum nodetype t, const char *s, struct node *np) 334 { 335 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 336 if (np->t != T_EVENT) 337 outfl(O_ERR, np->file, np->line, 338 "%s %s property must be an engine name " 339 "(i.e. stat.x or stat.x@a/b)", 340 ptree_nodetype2str(t), s); 341 342 /* XXX confirm engine has been declared */ 343 return (1); 344 } 345 346 static int 347 check_timeval(enum nodetype t, const char *s, struct node *np) 348 { 349 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 350 if (np->t != T_TIMEVAL) 351 outfl(O_ERR, np->file, np->line, 352 "%s %s property must be a number with time units", 353 ptree_nodetype2str(t), s); 354 return (1); 355 } 356 357 static int 358 check_id(enum nodetype t, const char *s, struct node *np) 359 { 360 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 361 if (np->t != T_NAME || np->u.name.next || np->u.name.child) 362 outfl(O_ERR, np->file, np->line, 363 "%s %s property must be simple name", 364 ptree_nodetype2str(t), s); 365 return (1); 366 } 367 368 static int 369 check_serd_method(enum nodetype t, const char *s, struct node *np) 370 { 371 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 372 if (np->t != T_NAME || np->u.name.next || np->u.name.child || 373 (np->u.name.s != L_volatile && 374 np->u.name.s != L_persistent)) 375 outfl(O_ERR, np->file, np->line, 376 "%s %s property must be \"volatile\" or \"persistent\"", 377 ptree_nodetype2str(t), s); 378 return (1); 379 } 380 381 static int 382 check_serd_id(enum nodetype t, const char *s, struct node *np) 383 { 384 ASSERTinfo(np != NULL, ptree_nodetype2str(t)); 385 if (np->t != T_GLOBID) 386 outfl(O_ERR, np->file, np->line, 387 "%s %s property must be a global ID", 388 ptree_nodetype2str(t), s); 389 return (1); 390 } 391 392 void 393 check_stmt_required_properties(struct node *stmtnp) 394 { 395 struct lut *lutp = stmtnp->u.stmt.lutp; 396 struct node *np = stmtnp->u.stmt.np; 397 int i; 398 399 for (i = 0; Allowednames[i].t; i++) 400 if (stmtnp->t == Allowednames[i].t && 401 Allowednames[i].required && 402 tree_s2np_lut_lookup(lutp, Allowednames[i].name) == NULL) 403 outfl(Allowednames[i].outflags, 404 np->file, np->line, 405 "%s statement missing property: %s", 406 ptree_nodetype2str(stmtnp->t), 407 Allowednames[i].name); 408 } 409 410 void 411 check_stmt_allowed_properties(enum nodetype t, 412 struct node *nvpairnp, struct lut *lutp) 413 { 414 int i; 415 const char *s = nvpairnp->u.expr.left->u.name.s; 416 struct node *np; 417 418 for (i = 0; Allowednames[i].t; i++) 419 if (t == Allowednames[i].t && Allowednames[i].name == NULL) { 420 /* NULL name means just call checker */ 421 (*Allowednames[i].checker)(t, s, 422 nvpairnp->u.expr.right); 423 return; 424 } else if (t == Allowednames[i].t && s == Allowednames[i].name) 425 break; 426 if (Allowednames[i].name == NULL) 427 outfl(O_ERR, nvpairnp->file, nvpairnp->line, 428 "illegal property name in %s declaration: %s", 429 ptree_nodetype2str(t), s); 430 else if ((np = tree_s2np_lut_lookup(lutp, s)) != NULL) { 431 /* 432 * redeclaring prop is allowed if value is the same 433 */ 434 if (np->t != nvpairnp->u.expr.right->t) 435 outfl(O_ERR, nvpairnp->file, nvpairnp->line, 436 "property redeclared (with differnt type) " 437 "in %s declaration: %s", 438 ptree_nodetype2str(t), s); 439 switch (np->t) { 440 case T_NUM: 441 case T_TIMEVAL: 442 if (np->u.ull == nvpairnp->u.expr.right->u.ull) 443 return; 444 break; 445 446 case T_NAME: 447 if (tree_namecmp(np, 448 nvpairnp->u.expr.right) == 0) 449 return; 450 break; 451 452 case T_EVENT: 453 if (tree_eventcmp(np, 454 nvpairnp->u.expr.right) == 0) 455 return; 456 break; 457 458 default: 459 outfl(O_ERR, nvpairnp->file, nvpairnp->line, 460 "value for property \"%s\" is an " 461 "invalid type: %s", 462 nvpairnp->u.expr.left->u.name.s, 463 ptree_nodetype2str(np->t)); 464 return; 465 } 466 outfl(O_ERR, nvpairnp->file, nvpairnp->line, 467 "property redeclared in %s declaration: %s", 468 ptree_nodetype2str(t), s); 469 } else 470 (*Allowednames[i].checker)(t, s, nvpairnp->u.expr.right); 471 } 472 473 void 474 check_propnames(enum nodetype t, struct node *np, int from, int to) 475 { 476 struct node *dnp; 477 struct lut *lutp; 478 479 ASSERT(np != NULL); 480 ASSERTinfo(np->t == T_EVENT || np->t == T_LIST || np->t == T_ARROW, 481 ptree_nodetype2str(np->t)); 482 483 if (np->t == T_EVENT) { 484 switch (np->u.event.ename->u.name.t) { 485 case N_UNSPEC: 486 outfl(O_ERR, np->file, np->line, 487 "name in %s statement must begin with " 488 "type (example: \"error.\")", 489 ptree_nodetype2str(t)); 490 return; 491 case N_FAULT: 492 lutp = Faults; 493 if (to) { 494 outfl(O_ERR, np->file, np->line, 495 "%s has fault on right side of \"->\"", 496 ptree_nodetype2str(t)); 497 return; 498 } 499 if (!from) { 500 outfl(O_DIE, np->file, np->line, 501 "internal error: %s has fault without " 502 "from flag", 503 ptree_nodetype2str(t)); 504 } 505 break; 506 case N_UPSET: 507 lutp = Upsets; 508 if (to) { 509 outfl(O_ERR, np->file, np->line, 510 "%s has upset on right side of \"->\"", 511 ptree_nodetype2str(t)); 512 return; 513 } 514 if (!from) 515 outfl(O_DIE, np->file, np->line, 516 "internal error: %s has upset without " 517 "from flag", 518 ptree_nodetype2str(t)); 519 break; 520 case N_DEFECT: 521 lutp = Defects; 522 if (to) { 523 outfl(O_ERR, np->file, np->line, 524 "%s has defect on right side of \"->\"", 525 ptree_nodetype2str(t)); 526 return; 527 } 528 if (!from) { 529 outfl(O_DIE, np->file, np->line, 530 "internal error: %s has defect without " 531 "from flag", 532 ptree_nodetype2str(t)); 533 } 534 break; 535 case N_ERROR: 536 lutp = Errors; 537 if (!from && !to) 538 outfl(O_DIE, np->file, np->line, 539 "%s has error without from or to flags", 540 ptree_nodetype2str(t)); 541 break; 542 case N_EREPORT: 543 lutp = Ereports; 544 if (from) { 545 outfl(O_ERR, np->file, np->line, 546 "%s has report on left side of \"->\"", 547 ptree_nodetype2str(t)); 548 return; 549 } 550 if (!to) 551 outfl(O_DIE, np->file, np->line, 552 "internal error: %s has report without " 553 "to flag", 554 ptree_nodetype2str(t)); 555 break; 556 default: 557 outfl(O_DIE, np->file, np->line, 558 "internal error: check_propnames: " 559 "unexpected type: %d", np->u.name.t); 560 } 561 562 if ((dnp = tree_event2np_lut_lookup(lutp, np)) == NULL) { 563 outfl(O_ERR, np->file, np->line, 564 "%s statement contains undeclared event", 565 ptree_nodetype2str(t)); 566 } else 567 dnp->u.stmt.flags |= STMT_REF; 568 np->u.event.declp = dnp; 569 } else if (np->t == T_LIST) { 570 check_propnames(t, np->u.expr.left, from, to); 571 check_propnames(t, np->u.expr.right, from, to); 572 } else if (np->t == T_ARROW) { 573 check_propnames(t, np->u.arrow.lhs, 1, to); 574 check_propnames(t, np->u.arrow.rhs, from, 1); 575 } 576 } 577 578 static struct lut * 579 record_iterators(struct node *np, struct lut *ex) 580 { 581 if (np == NULL) 582 return (ex); 583 584 switch (np->t) { 585 case T_ARROW: 586 ex = record_iterators(np->u.arrow.lhs, ex); 587 ex = record_iterators(np->u.arrow.rhs, ex); 588 break; 589 590 case T_LIST: 591 ex = record_iterators(np->u.expr.left, ex); 592 ex = record_iterators(np->u.expr.right, ex); 593 break; 594 595 case T_EVENT: 596 ex = record_iterators(np->u.event.epname, ex); 597 break; 598 599 case T_NAME: 600 if (np->u.name.child && np->u.name.child->t == T_NAME) 601 ex = lut_add(ex, (void *) np->u.name.child->u.name.s, 602 (void *) np, NULL); 603 ex = record_iterators(np->u.name.next, ex); 604 break; 605 606 default: 607 outfl(O_DIE, np->file, np->line, 608 "record_iterators: internal error: unexpected type: %s", 609 ptree_nodetype2str(np->t)); 610 } 611 612 return (ex); 613 } 614 615 void 616 check_exprscope(struct node *np, struct lut *ex) 617 { 618 if (np == NULL) 619 return; 620 621 switch (np->t) { 622 case T_EVENT: 623 check_exprscope(np->u.event.eexprlist, ex); 624 break; 625 626 case T_ARROW: 627 check_exprscope(np->u.arrow.lhs, ex); 628 check_exprscope(np->u.arrow.rhs, ex); 629 break; 630 631 case T_NAME: 632 if (np->u.name.child && np->u.name.child->t == T_NAME) { 633 if (lut_lookup(ex, 634 (void *) np->u.name.child->u.name.s, NULL) == NULL) 635 outfl(O_ERR, np->file, np->line, 636 "constraint contains undefined" 637 " iterator: %s", 638 np->u.name.child->u.name.s); 639 } 640 check_exprscope(np->u.name.next, ex); 641 break; 642 643 case T_QUOTE: 644 case T_GLOBID: 645 break; 646 647 case T_ASSIGN: 648 case T_NE: 649 case T_EQ: 650 case T_LIST: 651 case T_AND: 652 case T_OR: 653 case T_NOT: 654 case T_ADD: 655 case T_SUB: 656 case T_MUL: 657 case T_DIV: 658 case T_MOD: 659 case T_LT: 660 case T_LE: 661 case T_GT: 662 case T_GE: 663 case T_BITAND: 664 case T_BITOR: 665 case T_BITXOR: 666 case T_BITNOT: 667 case T_LSHIFT: 668 case T_RSHIFT: 669 case T_CONDIF: 670 case T_CONDELSE: 671 check_exprscope(np->u.expr.left, ex); 672 check_exprscope(np->u.expr.right, ex); 673 break; 674 675 case T_FUNC: 676 check_exprscope(np->u.func.arglist, ex); 677 break; 678 679 case T_NUM: 680 case T_TIMEVAL: 681 break; 682 683 default: 684 outfl(O_DIE, np->file, np->line, 685 "check_exprscope: internal error: unexpected type: %s", 686 ptree_nodetype2str(np->t)); 687 } 688 } 689 690 /* 691 * check_propscope -- check constraints for out of scope variable refs 692 */ 693 void 694 check_propscope(struct node *np) 695 { 696 struct lut *ex; 697 698 ex = record_iterators(np, NULL); 699 check_exprscope(np, ex); 700 lut_free(ex, NULL, NULL); 701 } 702 703 /* 704 * check_upset_engine -- validate the engine property in an upset statement 705 * 706 * we do this after the full parse tree has been constructed rather than while 707 * building the parse tree because it is inconvenient for the user if we 708 * require SERD engines to be declared before used in an upset "engine" 709 * property. 710 */ 711 712 /*ARGSUSED*/ 713 void 714 check_upset_engine(struct node *lhs, struct node *rhs, void *arg) 715 { 716 enum nodetype t = (enum nodetype)arg; 717 struct node *engnp; 718 struct node *declp; 719 720 ASSERTeq(rhs->t, t, ptree_nodetype2str); 721 722 if ((engnp = tree_s2np_lut_lookup(rhs->u.stmt.lutp, L_engine)) == NULL) 723 return; 724 725 ASSERT(engnp->t == T_EVENT); 726 727 if ((declp = tree_event2np_lut_lookup(SERDs, engnp)) == NULL) { 728 outfl(O_ERR, engnp->file, engnp->line, 729 "%s %s property contains undeclared name", 730 ptree_nodetype2str(t), L_engine); 731 return; 732 } 733 engnp->u.event.declp = declp; 734 } 735 736 /* 737 * check_refcount -- see if declared names are used 738 * 739 * this is run after the entire parse tree is constructed, so a refcount 740 * of zero means the name has been declared but otherwise not used. 741 */ 742 743 void 744 check_refcount(struct node *lhs, struct node *rhs, void *arg) 745 { 746 enum nodetype t = (enum nodetype)arg; 747 748 ASSERTeq(rhs->t, t, ptree_nodetype2str); 749 750 if (rhs->u.stmt.flags & STMT_REF) 751 return; 752 753 outfl(O_WARN|O_NONL, rhs->file, rhs->line, 754 "%s name declared but not used: ", ptree_nodetype2str(t)); 755 ptree_name(O_WARN|O_NONL, lhs); 756 out(O_WARN, NULL); 757 } 758 759 /* 760 * set check_cycle_warninglevel only for val >= 0 761 */ 762 int 763 check_cycle_level(long long val) 764 { 765 static int check_cycle_warninglevel = -1; 766 767 if (val == 0) 768 check_cycle_warninglevel = 0; 769 else if (val > 0) 770 check_cycle_warninglevel = 1; 771 772 return (check_cycle_warninglevel); 773 } 774 775 /* 776 * check_cycle -- see props from an error have cycles 777 * 778 * this is run after the entire parse tree is constructed, for 779 * each error that has been declared. 780 */ 781 782 /*ARGSUSED*/ 783 void 784 check_cycle(struct node *lhs, struct node *rhs, void *arg) 785 { 786 struct node *np; 787 788 ASSERTeq(rhs->t, T_ERROR, ptree_nodetype2str); 789 790 if (rhs->u.stmt.flags & STMT_CYCLE) 791 return; /* already reported this cycle */ 792 793 if (rhs->u.stmt.flags & STMT_CYMARK) { 794 #ifdef ESC 795 int warninglevel; 796 797 warninglevel = check_cycle_level(-1); 798 if (warninglevel <= 0) { 799 int olevel = O_ERR; 800 801 if (warninglevel == 0) 802 olevel = O_WARN; 803 804 out(olevel|O_NONL, "cycle in propagation tree: "); 805 ptree_name(olevel|O_NONL, rhs->u.stmt.np); 806 out(olevel, NULL); 807 } 808 #endif /* ESC */ 809 810 rhs->u.stmt.flags |= STMT_CYCLE; 811 } 812 813 rhs->u.stmt.flags |= STMT_CYMARK; 814 815 /* for each propagation */ 816 for (np = Props; np; np = np->u.stmt.next) 817 check_cycle_lhs(rhs, np->u.stmt.np); 818 819 rhs->u.stmt.flags &= ~STMT_CYMARK; 820 } 821 822 /* 823 * check_cycle_lhs -- find the lhs of an arrow for cycle checking 824 */ 825 826 static void 827 check_cycle_lhs(struct node *stmtnp, struct node *arrow) 828 { 829 struct node *trylhs; 830 struct node *tryrhs; 831 832 /* handle cascaded arrows */ 833 switch (arrow->u.arrow.lhs->t) { 834 case T_ARROW: 835 /* first recurse left */ 836 check_cycle_lhs(stmtnp, arrow->u.arrow.lhs); 837 838 /* 839 * return if there's a list of events internal to 840 * cascaded props (which is not allowed) 841 */ 842 if (arrow->u.arrow.lhs->u.arrow.rhs->t != T_EVENT) 843 return; 844 845 /* then try this arrow (thing cascaded *to*) */ 846 trylhs = arrow->u.arrow.lhs->u.arrow.rhs; 847 tryrhs = arrow->u.arrow.rhs; 848 break; 849 850 case T_EVENT: 851 case T_LIST: 852 trylhs = arrow->u.arrow.lhs; 853 tryrhs = arrow->u.arrow.rhs; 854 break; 855 856 default: 857 out(O_DIE, "lhs: unexpected type: %s", 858 ptree_nodetype2str(arrow->u.arrow.lhs->t)); 859 /*NOTREACHED*/ 860 } 861 862 check_cycle_lhs_try(stmtnp, trylhs, tryrhs); 863 } 864 865 /* 866 * check_cycle_lhs_try -- try matching an event name on lhs of an arrow 867 */ 868 869 static void 870 check_cycle_lhs_try(struct node *stmtnp, struct node *lhs, struct node *rhs) 871 { 872 if (lhs->t == T_LIST) { 873 check_cycle_lhs_try(stmtnp, lhs->u.expr.left, rhs); 874 check_cycle_lhs_try(stmtnp, lhs->u.expr.right, rhs); 875 return; 876 } 877 878 ASSERT(lhs->t == T_EVENT); 879 880 if (tree_eventcmp(stmtnp->u.stmt.np, lhs) != 0) 881 return; /* no match */ 882 883 check_cycle_rhs(rhs); 884 } 885 886 /* 887 * check_cycle_rhs -- foreach error on rhs, see if we cycle to a marked error 888 */ 889 890 static void 891 check_cycle_rhs(struct node *rhs) 892 { 893 struct node *dnp; 894 895 if (rhs->t == T_LIST) { 896 check_cycle_rhs(rhs->u.expr.left); 897 check_cycle_rhs(rhs->u.expr.right); 898 return; 899 } 900 901 ASSERT(rhs->t == T_EVENT); 902 903 if (rhs->u.event.ename->u.name.t != N_ERROR) 904 return; 905 906 if ((dnp = tree_event2np_lut_lookup(Errors, rhs)) == NULL) { 907 outfl(O_ERR|O_NONL, 908 rhs->file, rhs->line, 909 "unexpected undeclared event during cycle check"); 910 ptree_name(O_ERR|O_NONL, rhs); 911 out(O_ERR, NULL); 912 return; 913 } 914 check_cycle(NULL, dnp, 0); 915 } 916 917 /* 918 * Force iterators to be simple names, expressions, or numbers 919 */ 920 void 921 check_name_iterator(struct node *np) 922 { 923 if (np->u.name.child->t != T_NUM && 924 np->u.name.child->t != T_NAME && 925 np->u.name.child->t != T_CONDIF && 926 np->u.name.child->t != T_SUB && 927 np->u.name.child->t != T_ADD && 928 np->u.name.child->t != T_MUL && 929 np->u.name.child->t != T_DIV && 930 np->u.name.child->t != T_MOD && 931 np->u.name.child->t != T_LSHIFT && 932 np->u.name.child->t != T_RSHIFT) { 933 outfl(O_ERR|O_NONL, np->file, np->line, 934 "invalid iterator: "); 935 ptree_name_iter(O_ERR|O_NONL, np); 936 out(O_ERR, NULL); 937 } 938 } 939 940 /* 941 * Iterators on a declaration may only be implicit 942 */ 943 void 944 check_type_iterator(struct node *np) 945 { 946 while (np != NULL) { 947 if (np->t == T_EVENT) { 948 np = np->u.event.epname; 949 } else if (np->t == T_NAME) { 950 if (np->u.name.child != NULL && 951 np->u.name.child->t != T_NUM) { 952 outfl(O_ERR|O_NONL, np->file, np->line, 953 "explicit iterators disallowed " 954 "in declarations: "); 955 ptree_name_iter(O_ERR|O_NONL, np); 956 out(O_ERR, NULL); 957 } 958 np = np->u.name.next; 959 } else { 960 break; 961 } 962 } 963 } 964 965 void 966 check_func(struct node *np) 967 { 968 struct node *arglist = np->u.func.arglist; 969 970 ASSERTinfo(np->t == T_FUNC, ptree_nodetype2str(np->t)); 971 972 if (np->u.func.s == L_within) { 973 switch (arglist->t) { 974 case T_NUM: 975 if (arglist->u.ull != 0ULL) { 976 outfl(O_ERR, arglist->file, arglist->line, 977 "parameter of within must be 0" 978 ", \"infinity\" or a time value."); 979 } 980 break; 981 982 case T_NAME: 983 if (arglist->u.name.s != L_infinity) { 984 outfl(O_ERR, arglist->file, arglist->line, 985 "parameter of within must be 0" 986 ", \"infinity\" or a time value."); 987 } 988 break; 989 990 case T_LIST: 991 /* 992 * if two parameters, the left or min must be 993 * either T_NUM or T_TIMEVAL 994 */ 995 if (arglist->u.expr.left->t != T_NUM && 996 arglist->u.expr.left->t != T_TIMEVAL) { 997 outfl(O_ERR, arglist->file, arglist->line, 998 "first parameter of within must be" 999 " either a time value or zero."); 1000 } 1001 1002 /* 1003 * if two parameters, the right or max must 1004 * be either T_NUM, T_NAME or T_TIMEVAL 1005 */ 1006 if (arglist->u.expr.right->t != T_NUM && 1007 arglist->u.expr.right->t != T_TIMEVAL && 1008 arglist->u.expr.right->t != T_NAME) { 1009 outfl(O_ERR, arglist->file, arglist->line, 1010 "second parameter of within must " 1011 "be 0, \"infinity\" or time value."); 1012 } 1013 1014 /* 1015 * if right or left is a T_NUM it must 1016 * be zero 1017 */ 1018 if ((arglist->u.expr.left->t == T_NUM) && 1019 (arglist->u.expr.left->u.ull != 0ULL)) { 1020 outfl(O_ERR, arglist->file, arglist->line, 1021 "within parameter must be " 1022 "0 or a time value."); 1023 } 1024 if ((arglist->u.expr.right->t == T_NUM) && 1025 (arglist->u.expr.right->u.ull != 0ULL)) { 1026 outfl(O_ERR, arglist->file, arglist->line, 1027 "within parameter must be " 1028 "0 or a time value."); 1029 } 1030 1031 /* if right is a T_NAME it must be "infinity" */ 1032 if ((arglist->u.expr.right->t == T_NAME) && 1033 (arglist->u.expr.right->u.name.s != L_infinity)) { 1034 outfl(O_ERR, arglist->file, arglist->line, 1035 "\"infinity\" is the only " 1036 "valid name for within parameter."); 1037 } 1038 1039 /* 1040 * the first parameter [min] must not be greater 1041 * than the second parameter [max]. 1042 */ 1043 if (arglist->u.expr.left->u.ull > 1044 arglist->u.expr.right->u.ull) { 1045 outfl(O_ERR, arglist->file, arglist->line, 1046 "the first value (min) of" 1047 " within must be less than" 1048 " the second (max) value"); 1049 } 1050 break; 1051 1052 case T_TIMEVAL: 1053 break; /* no restrictions on T_TIMEVAL */ 1054 1055 default: 1056 outfl(O_ERR, arglist->file, arglist->line, 1057 "parameter of within must be 0" 1058 ", \"infinity\" or a time value."); 1059 } 1060 } else if (np->u.func.s == L_call) { 1061 if (arglist->t != T_QUOTE && 1062 arglist->t != T_LIST && 1063 arglist->t != T_GLOBID && 1064 arglist->t != T_CONDIF && 1065 arglist->t != T_LIST && 1066 arglist->t != T_FUNC) 1067 outfl(O_ERR, arglist->file, arglist->line, 1068 "invalid first argument to call()"); 1069 } else if (np->u.func.s == L_fru) { 1070 if (arglist->t != T_NAME) 1071 outfl(O_ERR, arglist->file, arglist->line, 1072 "argument to fru() must be a path"); 1073 } else if (np->u.func.s == L_asru) { 1074 if (arglist->t != T_NAME) 1075 outfl(O_ERR, arglist->file, arglist->line, 1076 "argument to asru() must be a path"); 1077 } else if (np->u.func.s == L_is_connected || 1078 np->u.func.s == L_is_under) { 1079 if (arglist->t == T_LIST && 1080 (arglist->u.expr.left->t == T_NAME || 1081 (arglist->u.expr.left->t == T_FUNC && 1082 (arglist->u.expr.left->u.func.s == L_fru || 1083 arglist->u.expr.left->u.func.s == L_asru))) && 1084 (arglist->u.expr.right->t == T_NAME || 1085 (arglist->u.expr.right->t == T_FUNC && 1086 (arglist->u.expr.right->u.func.s == L_fru || 1087 arglist->u.expr.right->u.func.s == L_asru)))) { 1088 if (arglist->u.expr.left->t == T_FUNC) 1089 check_func(arglist->u.expr.left); 1090 if (arglist->u.expr.right->t == T_FUNC) 1091 check_func(arglist->u.expr.right); 1092 } else { 1093 outfl(O_ERR, arglist->file, arglist->line, 1094 "%s() must have paths or calls to " 1095 "fru() and/or asru() as arguments", 1096 np->u.func.s); 1097 } 1098 } else if (np->u.func.s == L_is_on) { 1099 if (arglist->t == T_NAME || 1100 (arglist->t == T_FUNC && 1101 (arglist->u.func.s == L_fru || 1102 arglist->u.func.s == L_asru))) { 1103 if (arglist->t == T_FUNC) 1104 check_func(arglist); 1105 } else { 1106 outfl(O_ERR, arglist->file, arglist->line, 1107 "argument to is_on() must be a path or a call to " 1108 "fru() or asru()"); 1109 } 1110 } else if (np->u.func.s == L_is_present) { 1111 if (arglist->t == T_NAME || 1112 (arglist->t == T_FUNC && 1113 (arglist->u.func.s == L_fru || 1114 arglist->u.func.s == L_asru))) { 1115 if (arglist->t == T_FUNC) 1116 check_func(arglist); 1117 } else { 1118 outfl(O_ERR, arglist->file, arglist->line, 1119 "argument to is_present() must be a path or a call " 1120 "to fru() or asru()"); 1121 } 1122 } else if (np->u.func.s == L_is_type) { 1123 if (arglist->t == T_NAME || 1124 (arglist->t == T_FUNC && 1125 (arglist->u.func.s == L_fru || 1126 arglist->u.func.s == L_asru))) { 1127 if (arglist->t == T_FUNC) 1128 check_func(arglist); 1129 } else { 1130 outfl(O_ERR, arglist->file, arglist->line, 1131 "argument to is_type() must be a path or a call to " 1132 "fru() or asru()"); 1133 } 1134 } else if (np->u.func.s == L_confcall) { 1135 if (arglist->t != T_QUOTE && 1136 (arglist->t != T_LIST || 1137 arglist->u.expr.left->t != T_QUOTE)) 1138 outfl(O_ERR, arglist->file, arglist->line, 1139 "confcall(): first argument must be a string " 1140 "(the name of the operation)"); 1141 } else if (np->u.func.s == L_confprop || 1142 np->u.func.s == L_confprop_defined) { 1143 if (arglist->t == T_LIST && 1144 (arglist->u.expr.left->t == T_NAME || 1145 (arglist->u.expr.left->t == T_FUNC && 1146 (arglist->u.expr.left->u.func.s == L_fru || 1147 arglist->u.expr.left->u.func.s == L_asru))) && 1148 arglist->u.expr.right->t == T_QUOTE) { 1149 if (arglist->u.expr.left->t == T_FUNC) 1150 check_func(arglist->u.expr.left); 1151 } else { 1152 outfl(O_ERR, arglist->file, arglist->line, 1153 "%s(): first argument must be a path or a call to " 1154 "fru() or asru(); " 1155 "second argument must be a string", np->u.func.s); 1156 } 1157 } else if (np->u.func.s == L_count) { 1158 if (arglist->t != T_EVENT) { 1159 outfl(O_ERR, arglist->file, arglist->line, 1160 "count(): argument must be an engine name"); 1161 } 1162 } else if (np->u.func.s == L_defined) { 1163 if (arglist->t != T_GLOBID) 1164 outfl(O_ERR, arglist->file, arglist->line, 1165 "argument to defined() must be a global"); 1166 } else if (np->u.func.s == L_payloadprop) { 1167 if (arglist->t != T_QUOTE) 1168 outfl(O_ERR, arglist->file, arglist->line, 1169 "argument to payloadprop() must be a string"); 1170 } else if (np->u.func.s == L_payloadprop_contains) { 1171 if (arglist->t != T_LIST || 1172 arglist->u.expr.left->t != T_QUOTE || 1173 arglist->u.expr.right == NULL) 1174 outfl(O_ERR, arglist->file, arglist->line, 1175 "args to payloadprop_contains(): must be a quoted " 1176 "string (property name) and an expression " 1177 "(to match)"); 1178 } else if (np->u.func.s == L_payloadprop_defined) { 1179 if (arglist->t != T_QUOTE) 1180 outfl(O_ERR, arglist->file, arglist->line, 1181 "arg to payloadprop_defined(): must be a quoted " 1182 "string"); 1183 } else if (np->u.func.s == L_setpayloadprop) { 1184 if (arglist->t == T_LIST && 1185 arglist->u.expr.left->t == T_QUOTE) { 1186 if (arglist->u.expr.right->t == T_FUNC) 1187 check_func(arglist->u.expr.right); 1188 } else { 1189 outfl(O_ERR, arglist->file, arglist->line, 1190 "setpayloadprop(): " 1191 "first arg must be a string, " 1192 "second arg a value"); 1193 } 1194 } else if (np->u.func.s == L_setserdn || np->u.func.s == L_setserdt || 1195 np->u.func.s == L_setserdsuffix || np->u.func.s == 1196 L_setserdincrement) { 1197 if (arglist->t == T_FUNC) 1198 check_func(arglist); 1199 } else if (np->u.func.s == L_envprop) { 1200 if (arglist->t != T_QUOTE) 1201 outfl(O_ERR, arglist->file, arglist->line, 1202 "argument to envprop() must be a string"); 1203 } else 1204 outfl(O_WARN, np->file, np->line, 1205 "possible platform-specific function: %s", 1206 np->u.func.s); 1207 } 1208 1209 void 1210 check_expr(struct node *np) 1211 { 1212 ASSERT(np != NULL); 1213 1214 switch (np->t) { 1215 case T_ASSIGN: 1216 ASSERT(np->u.expr.left != NULL); 1217 if (np->u.expr.left->t != T_GLOBID) 1218 outfl(O_ERR, np->file, np->line, 1219 "assignment only allowed to globals (e.g. $a)"); 1220 break; 1221 } 1222 } 1223 1224 void 1225 check_event(struct node *np) 1226 { 1227 ASSERT(np != NULL); 1228 ASSERTinfo(np->t == T_EVENT, ptree_nodetype2str(np->t)); 1229 1230 if (np->u.event.epname == NULL) { 1231 outfl(O_ERR|O_NONL, np->file, np->line, 1232 "pathless events not allowed: "); 1233 ptree_name(O_ERR|O_NONL, np->u.event.ename); 1234 out(O_ERR, NULL); 1235 } 1236 } 1237 1238 /* 1239 * check for properties that are required on declarations. This 1240 * should be done after all declarations since they can be 1241 * redeclared with a different set of properties. 1242 */ 1243 /*ARGSUSED*/ 1244 void 1245 check_required_props(struct node *lhs, struct node *rhs, void *arg) 1246 { 1247 ASSERTeq(rhs->t, (enum nodetype)arg, ptree_nodetype2str); 1248 1249 check_stmt_required_properties(rhs); 1250 } 1251 1252 /* 1253 * check that cascading prop statements do not contain lists internally. 1254 * the first and last event lists in the cascading prop may be single 1255 * events or lists of events. 1256 */ 1257 /*ARGSUSED*/ 1258 void 1259 check_proplists(enum nodetype t, struct node *np) 1260 { 1261 ASSERT(np->t == T_ARROW); 1262 /* 1263 * not checking the right hand side of the top level prop 1264 * since it is the last part of the propagation and can be 1265 * an event or list of events 1266 */ 1267 check_proplists_lhs(t, np->u.arrow.lhs); 1268 } 1269 1270 /*ARGSUSED*/ 1271 static void 1272 check_proplists_lhs(enum nodetype t, struct node *lhs) 1273 { 1274 if (lhs->t == T_ARROW) { 1275 if (lhs->u.arrow.rhs->t == T_LIST) { 1276 outfl(O_ERR, lhs->file, lhs->line, 1277 "lists are not allowed internally on cascading %s", 1278 (t == T_PROP) ? "propagations" : "masks"); 1279 } 1280 check_proplists_lhs(t, lhs->u.arrow.lhs); 1281 } 1282 } 1283