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 /* 23 * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 27 /* 28 * Map file parsing, Version 2 syntax (solaris). 29 */ 30 #include <stdio.h> 31 #include <unistd.h> 32 #include <ctype.h> 33 #include <sys/elf_amd64.h> /* SHF_AMD64_LARGE */ 34 #include <elfcap.h> 35 #include "msg.h" 36 #include "_libld.h" 37 #include "_map.h" 38 39 /* 40 * Use a case insensitive string match when looking up capability mask 41 * values by name, and omit the AV_ prefix. 42 */ 43 #define ELFCAP_STYLE ELFCAP_STYLE_LC | ELFCAP_STYLE_F_ICMP 44 45 /* 46 * Signature for functions used to parse top level mapfile directives 47 */ 48 typedef Token (*dir_func_t)(Mapfile *mf); 49 50 /* 51 * Signature for functions used to parse attribute level assignments 52 * mf - Mapfile descriptor 53 * eq_tok - One of the equal tokens (TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ) 54 * or TK_ERROR. See the comment for attr_fmt_t below. 55 * uvalue - An arbitrary pointer "user value" passed by the 56 * caller to parse_attributes() for use by the function. 57 */ 58 typedef Token (* attr_func_t)(Mapfile *mf, Token eq_tok, void *uvalue); 59 60 /* 61 * Signature for gettoken_str() err_func argument. This is a function 62 * called to issue an appropriate error message. 63 * 64 * The gts prefix stands for "Get Token Str" 65 */ 66 typedef void (* gts_efunc_t)(Mapfile *mf, Token tok, ld_map_tkval_t *tkv); 67 68 /* 69 * The attr_fmt_t tells parse_attributes how far to go in parsing 70 * an attribute before it calls the at_func function to take over: 71 * 72 * ATTR_FMT_NAME - Parse the name, and immediately call the function. 73 * This is useful in cases where there is more than 74 * one possible syntax for a given attribute. The value of 75 * eq_tok passed to the at_func function will be TK_ERROR, 76 * reflecting the fact that it has no meaning in this context. 77 * 78 * ATTR_FMT_EQ - Parse the name, and the following '=', and then call 79 * the function. The value passed to the at_func function for 80 * eq_tok will be TK_EQUAL. 81 * 82 * ATTR_FMT_EQ_PEQ - Parse the name, and a following equal token which 83 * can be '=' or '+=', and then call the function. The value 84 * passed to the at_func function for eq_tok will be one of 85 * TK_EQUAL, or TK_PLUSEQ. 86 * 87 * ATTR_FMT_EQ_ALL - Parse the name, and a following equal token which 88 * can be any of the three forms (=, +=, -=), and then call 89 * the function. The value passed to the at_func function for 90 * eq_tok will be one of TK_EQUAL, TK_PLUSEQ, or TK_MINUSEQ. 91 */ 92 typedef enum { 93 ATTR_FMT_NAME, 94 ATTR_FMT_EQ, 95 ATTR_FMT_EQ_PEQ, 96 ATTR_FMT_EQ_ALL, 97 } attr_fmt_t; 98 99 /* 100 * Type used to describe a set of valid attributes to parse_attributes(): 101 * at_name - Name of attribute 102 * at_func - Function to call when attribute is recognized, 103 * at_all_eq - True if attribute allows the '+=' and '-=' forms of 104 * assignment token, and False to only allow '='. 105 * 106 * The array of these structs passed to parse_attributes() must be 107 * NULL terminated (the at_name field must be set to NULL). 108 */ 109 typedef struct { 110 const char *at_name; /* Name of attribute */ 111 attr_func_t at_func; /* Function to call */ 112 attr_fmt_t at_fmt; /* How much to parse before calling */ 113 /* at_func */ 114 } attr_t; 115 116 /* 117 * Mapfile version and symbol state are separate but related concepts 118 * that are best represented using two different types. However, our 119 * style of passing a single uvalue via parse_attributes() makes it 120 * convenient to be able to reference them from a single address. 121 */ 122 typedef struct { 123 ld_map_ver_t ss_mv; 124 ld_map_sym_t ss_ms; 125 } symbol_state_t; 126 127 /* 128 * Process an expected equal operator. Deals with the fact that we 129 * have three variants. 130 * 131 * entry: 132 * mf - Mapfile descriptor 133 * eq_type - Types of equal operators accepted. One of ATTR_FMT_EQ, 134 * ATTR_FMT_EQ_PEQ, or ATTR_FMT_EQ_ALL. 135 * lhs - Name that appears on the left hand side of the expected 136 * equal operator. 137 * 138 * exit: 139 * Returns one of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, or TK_ERROR. 140 */ 141 static Token 142 gettoken_eq(Mapfile *mf, attr_fmt_t eq_type, const char *lhs) 143 { 144 Token tok; 145 ld_map_tkval_t tkv; 146 const char *err; 147 Conv_inv_buf_t inv_buf; 148 149 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 150 case TK_ERROR: 151 case TK_EQUAL: 152 return (tok); 153 154 case TK_PLUSEQ: 155 switch (eq_type) { 156 case ATTR_FMT_EQ_PEQ: 157 case ATTR_FMT_EQ_ALL: 158 return (tok); 159 } 160 break; 161 162 case TK_MINUSEQ: 163 if (eq_type == ATTR_FMT_EQ_ALL) 164 return (tok); 165 break; 166 } 167 168 switch (eq_type) { 169 case ATTR_FMT_EQ: 170 err = MSG_INTL(MSG_MAP_EXP_EQ); 171 break; 172 case ATTR_FMT_EQ_PEQ: 173 err = MSG_INTL(MSG_MAP_EXP_EQ_PEQ); 174 break; 175 case ATTR_FMT_EQ_ALL: 176 err = MSG_INTL(MSG_MAP_EXP_EQ_ALL); 177 break; 178 default: 179 /*NOTREACHED*/ 180 assert(0); 181 } 182 mf_fatal(mf, err, lhs, ld_map_tokenstr(tok, &tkv, &inv_buf)); 183 return (TK_ERROR); 184 } 185 186 /* 187 * Apply one of the three equal tokens to a bitmask value 188 * 189 * entry: 190 * dst - Address of bitmask variable to alter 191 * eq_tok - One of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, representing 192 * the operation to carry out. 193 * value - Value for right hand side 194 * 195 * exit: 196 * The operation has been carried out: 197 * 198 * TK_EQUAL - *dst is set to value 199 * TK_PLUSEQ - Bits in value have been set in *dst 200 * TK_MINUSEQ - Bits in value have been removed from *dst 201 */ 202 static void 203 setflags_eq(Word *dst, Token eq_tok, Word value) 204 { 205 switch (eq_tok) { 206 case TK_EQUAL: 207 *dst = value; 208 break; 209 case TK_PLUSEQ: 210 *dst |= value; 211 break; 212 case TK_MINUSEQ: 213 *dst &= ~value; 214 break; 215 default: 216 /*NOTREACHED*/ 217 assert(0); 218 } 219 } 220 221 /* 222 * Apply one of the three equal tokens to a capabilities Capmask. 223 * 224 * entry: 225 * mf - Mapfile descriptor 226 * capmask - Address of Capmask variable to alter 227 * eq_tok - One of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, representing 228 * the operation to carry out. 229 * type - Capability type (CA_SUNW_*) 230 * value - Value for right hand side 231 * title - True if a title is needed, False otherwise. 232 * 233 * exit: 234 * On success, returns TRUE (1), otherwise FALSE (0) 235 */ 236 static Boolean 237 set_capmask(Mapfile *mf, Capmask *capmask, Token eq_tok, 238 Word type, elfcap_mask_t value, Boolean title) 239 { 240 if (title) 241 DBG_CALL(Dbg_cap_mapfile_title(mf->mf_ofl->ofl_lml, 242 mf->mf_lineno)); 243 DBG_CALL(Dbg_cap_val_entry(mf->mf_ofl->ofl_lml, DBG_STATE_CURRENT, 244 type, capmask->cm_val, ld_targ.t_m.m_mach)); 245 246 switch (eq_tok) { 247 case TK_EQUAL: 248 capmask->cm_val = value; 249 capmask->cm_exc = 0; 250 ld_map_cap_set_ovflag(mf, type); 251 DBG_CALL(Dbg_cap_val_entry(mf->mf_ofl->ofl_lml, 252 DBG_STATE_RESET, type, capmask->cm_val, 253 ld_targ.t_m.m_mach)); 254 break; 255 case TK_PLUSEQ: 256 DBG_CALL(Dbg_cap_val_entry(mf->mf_ofl->ofl_lml, 257 DBG_STATE_ADD, type, value, ld_targ.t_m.m_mach)); 258 capmask->cm_val |= value; 259 capmask->cm_exc &= ~value; 260 break; 261 case TK_MINUSEQ: 262 DBG_CALL(Dbg_cap_val_entry(mf->mf_ofl->ofl_lml, 263 DBG_STATE_EXCLUDE, type, value, ld_targ.t_m.m_mach)); 264 capmask->cm_val &= ~value; 265 capmask->cm_exc |= value; 266 break; 267 default: 268 /*NOTREACHED*/ 269 assert(0); 270 } 271 272 /* Sanity check the resulting bits */ 273 if (!ld_map_cap_sanitize(mf, type, capmask)) 274 return (FALSE); 275 276 /* Report the final configuration */ 277 DBG_CALL(Dbg_cap_val_entry(mf->mf_ofl->ofl_lml, 278 DBG_STATE_RESOLVED, type, capmask->cm_val, ld_targ.t_m.m_mach)); 279 280 return (TRUE); 281 } 282 283 /* 284 * Apply one of the three equal tokens to a capabilities Caplist. 285 * 286 * entry: 287 * mf - Mapfile descriptor 288 * caplist - Address of Caplist variable to alter 289 * eq_tok - One of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, representing 290 * the operation to carry out. 291 * type - Capability type (CA_SUNW_*) 292 * str - String for right hand side 293 * title - True if a title is needed, False otherwise. 294 * 295 * exit: 296 * On success, returns TRUE (1), otherwise FALSE (0) 297 */ 298 static Boolean 299 set_capstr(Mapfile *mf, Caplist *caplist, Token eq_tok, 300 Word type, APlist *strs) 301 { 302 Capstr *capstr; 303 Aliste idx1; 304 char *str; 305 306 DBG_CALL(Dbg_cap_mapfile_title(mf->mf_ofl->ofl_lml, mf->mf_lineno)); 307 308 if ((caplist->cl_val == NULL) || (alist_nitems(caplist->cl_val) == 0)) { 309 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 310 DBG_STATE_CURRENT, type, NULL)); 311 } else { 312 for (ALIST_TRAVERSE(caplist->cl_val, idx1, capstr)) { 313 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 314 DBG_STATE_CURRENT, type, capstr->cs_str)); 315 } 316 } 317 318 switch (eq_tok) { 319 case TK_EQUAL: 320 if (caplist->cl_val) { 321 (void) free(caplist->cl_val); 322 caplist->cl_val = NULL; 323 } 324 if (caplist->cl_exc) { 325 (void) free(caplist->cl_exc); 326 caplist->cl_exc = NULL; 327 } 328 if (strs) { 329 for (APLIST_TRAVERSE(strs, idx1, str)) { 330 if ((capstr = alist_append(&caplist->cl_val, 331 NULL, sizeof (Capstr), 332 AL_CNT_CAP_NAMES)) == NULL) 333 return (FALSE); 334 capstr->cs_str = str; 335 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 336 DBG_STATE_RESET, type, capstr->cs_str)); 337 } 338 } else { 339 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 340 DBG_STATE_RESET, type, NULL)); 341 } 342 ld_map_cap_set_ovflag(mf, type); 343 break; 344 case TK_PLUSEQ: 345 for (APLIST_TRAVERSE(strs, idx1, str)) { 346 Aliste idx2; 347 const char *ostr; 348 int found = 0; 349 350 /* 351 * Add this name to the list of names, provided the 352 * name doesn't already exist. 353 */ 354 for (ALIST_TRAVERSE(caplist->cl_val, idx2, capstr)) { 355 if (strcmp(str, capstr->cs_str) == 0) { 356 found++; 357 break; 358 } 359 } 360 if ((found == 0) && ((capstr = 361 (Capstr *)alist_append(&caplist->cl_val, NULL, 362 sizeof (Capstr), AL_CNT_CAP_NAMES)) == NULL)) 363 return (FALSE); 364 capstr->cs_str = str; 365 366 /* 367 * Remove this name from the list of excluded names, 368 * provided the name already exists. 369 */ 370 for (APLIST_TRAVERSE(caplist->cl_exc, idx2, ostr)) { 371 if (strcmp(str, ostr) == 0) { 372 aplist_delete(caplist->cl_exc, &idx2); 373 break; 374 } 375 } 376 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 377 DBG_STATE_ADD, type, str)); 378 } 379 break; 380 case TK_MINUSEQ: 381 for (APLIST_TRAVERSE(strs, idx1, str)) { 382 Aliste idx2; 383 const char *ostr; 384 int found = 0; 385 386 /* 387 * Delete this name from the list of names, provided 388 * the name already exists. 389 */ 390 for (ALIST_TRAVERSE(caplist->cl_val, idx2, capstr)) { 391 if (strcmp(str, capstr->cs_str) == 0) { 392 alist_delete(caplist->cl_val, &idx2); 393 break; 394 } 395 } 396 397 /* 398 * Add this name to the list of excluded names, 399 * provided the name already exists. 400 */ 401 for (APLIST_TRAVERSE(caplist->cl_exc, idx2, ostr)) { 402 if (strcmp(str, ostr) == 0) { 403 found++; 404 break; 405 } 406 } 407 if ((found == 0) && (aplist_append(&caplist->cl_exc, 408 str, AL_CNT_CAP_NAMES) == NULL)) 409 return (FALSE); 410 411 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 412 DBG_STATE_EXCLUDE, type, str)); 413 } 414 break; 415 default: 416 /*NOTREACHED*/ 417 assert(0); 418 } 419 420 /* Report the final configuration */ 421 if ((caplist->cl_val == NULL) || (alist_nitems(caplist->cl_val) == 0)) { 422 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 423 DBG_STATE_RESOLVED, type, NULL)); 424 } else { 425 for (ALIST_TRAVERSE(caplist->cl_val, idx1, capstr)) { 426 DBG_CALL(Dbg_cap_ptr_entry(mf->mf_ofl->ofl_lml, 427 DBG_STATE_RESOLVED, type, capstr->cs_str)); 428 } 429 } 430 431 return (TRUE); 432 } 433 434 /* 435 * Process the next token, which is expected to start an optional 436 * nesting of attributes (';' or '{'). 437 * 438 * entry: 439 * mf - Mapfile descriptor 440 * lhs - Name of the directive or attribute being processed. 441 * 442 * exit: 443 * Returns TK_SEMICOLON or TK_LEFTBKT for success, and TK_ERROR otherwise. 444 */ 445 static Token 446 gettoken_optattr(Mapfile *mf, const char *lhs) 447 { 448 Token tok; 449 ld_map_tkval_t tkv; 450 Conv_inv_buf_t inv_buf; 451 452 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 453 case TK_ERROR: 454 case TK_SEMICOLON: 455 case TK_LEFTBKT: 456 return (tok); 457 } 458 459 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEMLBKT), lhs, 460 ld_map_tokenstr(tok, &tkv, &inv_buf)); 461 return (TK_ERROR); 462 } 463 464 /* 465 * Process the next token, which is expected to be a line terminator 466 * (';' or '}'). 467 * 468 * entry: 469 * mf - Mapfile descriptor 470 * lhs - Name of the directive or attribute being processed. 471 * 472 * exit: 473 * Returns TK_SEMICOLON or TK_RIGHTBKT for success, and TK_ERROR otherwise. 474 */ 475 static Token 476 gettoken_term(Mapfile *mf, const char *lhs) 477 { 478 Token tok; 479 ld_map_tkval_t tkv; 480 Conv_inv_buf_t inv_buf; 481 482 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 483 case TK_ERROR: 484 case TK_SEMICOLON: 485 case TK_RIGHTBKT: 486 return (tok); 487 } 488 489 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEMRBKT), lhs, 490 ld_map_tokenstr(tok, &tkv, &inv_buf)); 491 return (TK_ERROR); 492 } 493 494 /* 495 * Process the next token, which is expected to be a semicolon. 496 * 497 * entry: 498 * mf - Mapfile descriptor 499 * lhs - Name of the directive or attribute being processed. 500 * 501 * exit: 502 * Returns TK_SEMICOLON for success, and TK_ERROR otherwise. 503 */ 504 static Token 505 gettoken_semicolon(Mapfile *mf, const char *lhs) 506 { 507 Token tok; 508 ld_map_tkval_t tkv; 509 Conv_inv_buf_t inv_buf; 510 511 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 512 case TK_ERROR: 513 case TK_SEMICOLON: 514 return (tok); 515 } 516 517 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEM), lhs, 518 ld_map_tokenstr(tok, &tkv, &inv_buf)); 519 return (TK_ERROR); 520 } 521 522 /* 523 * Process the next token, which is expected to be a '{' 524 * 525 * entry: 526 * mf - Mapfile descriptor 527 * lhs - Name of the item directly to the left of the expected left 528 * bracket. 529 * 530 * exit: 531 * Returns TK_LEFTBKT for success, and TK_ERROR otherwise. 532 */ 533 static Token 534 gettoken_leftbkt(Mapfile *mf, const char *lhs) 535 { 536 Token tok; 537 ld_map_tkval_t tkv; 538 Conv_inv_buf_t inv_buf; 539 540 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 541 case TK_ERROR: 542 case TK_LEFTBKT: 543 return (tok); 544 } 545 546 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_LBKT), lhs, 547 ld_map_tokenstr(tok, &tkv, &inv_buf)); 548 return (TK_ERROR); 549 } 550 551 /* 552 * Process the next token, which is expected to be an integer 553 * 554 * entry: 555 * mf - Mapfile descriptor 556 * lhs - Name of the directive or attribute being processed. 557 * tkv - Address of token value struct to be filled in 558 * 559 * exit: 560 * Updates *tkv and returns TK_INT for success, TK_ERROR otherwise. 561 */ 562 static Token 563 gettoken_int(Mapfile *mf, const char *lhs, ld_map_tkval_t *tkv) 564 { 565 Token tok; 566 Conv_inv_buf_t inv_buf; 567 568 switch (tok = ld_map_gettoken(mf, 0, tkv)) { 569 case TK_ERROR: 570 case TK_INT: 571 return (tok); 572 } 573 574 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_INT), lhs, 575 ld_map_tokenstr(tok, tkv, &inv_buf)); 576 return (TK_ERROR); 577 } 578 579 /* 580 * Process the next token, which is expected to be a string 581 * 582 * entry: 583 * mf - Mapfile descriptor 584 * lhs - Name of the directive or attribute being processed. 585 * tkv - Address of token value struct to be filled in 586 * err_func - Function to call if an error occurs 587 * 588 * exit: 589 * Updates *tkv and returns TK_STRING for success. Calls the 590 * supplied err_func function and returns TK_ERROR otherwise. 591 */ 592 static Token 593 gettoken_str(Mapfile *mf, int flags, ld_map_tkval_t *tkv, gts_efunc_t efunc) 594 { 595 Token tok; 596 597 switch (tok = ld_map_gettoken(mf, flags, tkv)) { 598 case TK_ERROR: 599 case TK_STRING: 600 return (tok); 601 } 602 603 /* User supplied function reports the error */ 604 (* efunc)(mf, tok, tkv); 605 606 return (TK_ERROR); 607 } 608 609 /* 610 * Given a construct of the following common form: 611 * 612 * item_name { 613 * attribute = ...; 614 * ... 615 * } 616 * 617 * where the caller has detected the item_name and opening bracket, 618 * parse the construct and call the attribute functions for each 619 * attribute detected, stopping when the closing '}' is seen. 620 * 621 * entry: 622 * mf - Mapfile descriptor 623 * item_name - Already detected name of item for which attributes 624 * are being parsed. 625 * attr_list - NULL terminated array of attr_t structures describing the 626 * valid attributes for the item. 627 * expect_str - Comma separated string listing the names of expected 628 * attributes. 629 * uvalue - User value, passed to the attribute functions without 630 * examination by parse_attributes(), usable for maintaining 631 * shared state between the caller and the functions. 632 * 633 * exit: 634 * parse_attributes() reads the attribute name and equality token, 635 * and then calls the attribute function given by the attr_list array 636 * to handle everything up to and including the terminating ';'. 637 * This continues until the closing '}' is seen. 638 * 639 * If everything is successful, TK_RIGHTBKT is returned. Otherwise, 640 * a suitable error is issued and TK_ERROR is returned. 641 */ 642 static Token 643 parse_attributes(Mapfile *mf, const char *item_name, attr_t *attr_list, 644 size_t attr_list_bufsize, void *uvalue) 645 { 646 attr_t *attr; 647 Token tok, op_tok; 648 ld_map_tkval_t tkv; 649 int done; 650 int attr_cnt = 0; 651 Conv_inv_buf_t inv_buf; 652 653 /* Read attributes until the closing '}' is seen */ 654 for (done = 0; done == 0; ) { 655 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 656 case TK_ERROR: 657 return (TK_ERROR); 658 659 case TK_STRING: 660 attr = ld_map_kwfind(tkv.tkv_str, attr_list, 661 SGSOFFSETOF(attr_t, at_name), sizeof (attr[0])); 662 if (attr == NULL) 663 goto bad_attr; 664 665 /* 666 * Depending on the value of at_fmt, there are 667 * fout different actions to take: 668 * ATTR_FMT_NAME - Call at_func function 669 * ATTR_FMT_EQ - Read and verify a TK_EQUAL 670 * ATTR_FMT_EQ_PEQ - Read and verify a TK_EQUAL 671 * or TK_PLUSEQ. 672 * ATTR_FMT_EQ_ALL - Read/Verify one of the 673 * three possible equal tokens 674 * (TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ). 675 */ 676 if (attr->at_fmt == ATTR_FMT_NAME) { 677 /* Arbitrary value to pass to at_func */ 678 op_tok = TK_ERROR; 679 } else { 680 /* Read/Verify appropriate equal operator */ 681 op_tok = gettoken_eq(mf, attr->at_fmt, 682 attr->at_name); 683 if (op_tok == TK_ERROR) 684 return (TK_ERROR); 685 } 686 687 /* Call the associated function */ 688 switch (tok = attr->at_func(mf, op_tok, uvalue)) { 689 default: 690 return (TK_ERROR); 691 case TK_SEMICOLON: 692 break; 693 case TK_RIGHTBKT: 694 done = 1; 695 break; 696 } 697 attr_cnt++; 698 break; 699 700 case TK_RIGHTBKT: 701 done = 1; 702 break; 703 704 case TK_SEMICOLON: 705 break; /* Ignore empty statement */ 706 707 default: 708 bad_attr: 709 { 710 char buf[VLA_SIZE(attr_list_bufsize)]; 711 712 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_ATTR), 713 ld_map_kwnames(attr_list, 714 SGSOFFSETOF(attr_t, at_name), 715 sizeof (attr[0]), buf, attr_list_bufsize), 716 ld_map_tokenstr(tok, &tkv, &inv_buf)); 717 } 718 return (TK_ERROR); 719 } 720 } 721 722 /* Make sure there was at least one attribute between the {} brackets */ 723 if (attr_cnt == 0) { 724 mf_fatal(mf, MSG_INTL(MSG_MAP_NOATTR), item_name); 725 return (TK_ERROR); 726 } 727 728 return (tok); 729 } 730 731 /* 732 * Read whitespace delimited segment flags from the input and convert into 733 * bitmask of PF_ values they represent. Flags are terminated by a semicolon 734 * or right bracket. 735 * 736 * entry: 737 * mf - Mapfile descriptor 738 * flags - Address of variable to be set to resulting flags value 739 * 740 * exit: 741 * Returns the terminator token (TK_SEMICOLON or TK_LEFTBKT) on success, 742 * and TK_ERROR otherwise. 743 */ 744 static Token 745 parse_segment_flags(Mapfile *mf, Xword *flags) 746 { 747 /* 748 * Map flag names to their values. Since DATA and STACK have 749 * platform dependent values, we have to determine them at runtime. 750 * We indicate this by setting the top bit. 751 */ 752 #define PF_DATA 0x80000000 753 #define PF_STACK 0x80000001 754 typedef struct { 755 const char *name; 756 Word value; 757 } segflag_t; 758 static segflag_t flag_list[] = { 759 { MSG_ORIG(MSG_MAPKW_DATA), PF_DATA }, 760 { MSG_ORIG(MSG_MAPKW_EXECUTE), PF_X }, 761 { MSG_ORIG(MSG_MAPKW_READ), PF_R }, 762 { MSG_ORIG(MSG_MAPKW_STACK), PF_STACK }, 763 { MSG_ORIG(MSG_MAPKW_WRITE), PF_W }, 764 765 /* List must be null terminated */ 766 { 0 }, 767 }; 768 769 /* 770 * Size of buffer needed to format the names in flag_list[]. Must 771 * be kept in sync with flag_list. 772 */ 773 static size_t flag_list_bufsize = 774 KW_NAME_SIZE(MSG_MAPKW_DATA) + 775 KW_NAME_SIZE(MSG_MAPKW_EXECUTE) + 776 KW_NAME_SIZE(MSG_MAPKW_READ) + 777 KW_NAME_SIZE(MSG_MAPKW_STACK) + 778 KW_NAME_SIZE(MSG_MAPKW_WRITE); 779 780 Token tok; 781 ld_map_tkval_t tkv; 782 segflag_t *flag; 783 size_t cnt = 0; 784 int done; 785 Conv_inv_buf_t inv_buf; 786 787 *flags = 0; 788 789 /* Read attributes until the ';' terminator is seen */ 790 for (done = 0; done == 0; ) { 791 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 792 case TK_ERROR: 793 return (TK_ERROR); 794 795 case TK_STRING: 796 flag = ld_map_kwfind(tkv.tkv_str, flag_list, 797 SGSOFFSETOF(segflag_t, name), 798 sizeof (flag_list[0])); 799 if (flag == NULL) 800 goto bad_flag; 801 switch (flag->value) { 802 case PF_DATA: 803 *flags |= ld_targ.t_m.m_dataseg_perm; 804 break; 805 case PF_STACK: 806 *flags |= ld_targ.t_m.m_stack_perm; 807 break; 808 default: 809 *flags |= flag->value; 810 } 811 cnt++; 812 break; 813 814 case TK_INT: 815 /* 816 * Accept 0 for notational convenience, but refuse 817 * any other value. Note that we don't actually have 818 * to set the flags to 0 here, because there are 819 * already initialized to that before the main loop. 820 */ 821 if (tkv.tkv_int.tkvi_value != 0) 822 goto bad_flag; 823 cnt++; 824 break; 825 826 case TK_SEMICOLON: 827 case TK_RIGHTBKT: 828 done = 1; 829 break; 830 831 default: 832 bad_flag: 833 { 834 char buf[VLA_SIZE(flag_list_bufsize)]; 835 836 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEGFLAG), 837 ld_map_kwnames(flag_list, 838 SGSOFFSETOF(segflag_t, name), 839 sizeof (flag[0]), buf, flag_list_bufsize), 840 ld_map_tokenstr(tok, &tkv, &inv_buf)); 841 } 842 return (TK_ERROR); 843 } 844 } 845 846 /* Make sure there was at least one flag */ 847 if (cnt == 0) { 848 mf_fatal(mf, MSG_INTL(MSG_MAP_NOVALUES), 849 MSG_ORIG(MSG_MAPKW_FLAGS)); 850 return (TK_ERROR); 851 } 852 853 return (tok); 854 855 #undef PF_DATA 856 #undef PF_STACK 857 } 858 859 /* 860 * Parse one of the capabilities attributes that corresponds directly to a 861 * capabilities bitmask value (CA_SUNW_HW_x, CA_SUNW_SF_xx). Values can be 862 * integers, or symbolic names that correspond to the capabilities mask 863 * in question. 864 * 865 * entry: 866 * mf - Mapfile descriptor 867 * eq_tok - One of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, representing 868 * the operation to carry out. 869 * capmask - Capmask from output descriptor for capability being processed. 870 * type - Capability type (CA_SUNW_*) 871 * elfcap_from_str_func - pointer to elfcap-string-to-value function 872 * for capability being processed. 873 * 874 * exit: 875 * Returns TK_SEMICOLON or TK_RIGHTBKT for success, and TK_ERROR otherwise. 876 */ 877 static Token 878 parse_cap_mask(Mapfile *mf, Token eq_tok, Capmask *capmask, 879 Word type, elfcap_from_str_func_t *elfcap_from_str_func) 880 { 881 int done; 882 Token tok; 883 ld_map_tkval_t tkv; 884 Conv_inv_buf_t inv_buf; 885 elfcap_mask_t value = 0; 886 uint64_t v; 887 888 for (done = 0; done == 0; ) { 889 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 890 case TK_ERROR: 891 return (TK_ERROR); 892 893 case TK_STRING: 894 if ((v = (* elfcap_from_str_func)(ELFCAP_STYLE, 895 tkv.tkv_str, ld_targ.t_m.m_mach)) != 0) { 896 value |= v; 897 break; 898 } 899 goto bad_flag; 900 901 case TK_INT: 902 value |= tkv.tkv_int.tkvi_value; 903 break; 904 905 case TK_SEMICOLON: 906 case TK_RIGHTBKT: 907 done = 1; 908 break; 909 910 default: 911 bad_flag: 912 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_CAPMASK), 913 ld_map_tokenstr(tok, &tkv, &inv_buf)); 914 return (TK_ERROR); 915 } 916 } 917 918 if (!set_capmask(mf, capmask, eq_tok, type, value, TRUE)) 919 return (TK_ERROR); 920 return (tok); 921 } 922 923 /* 924 * Parse one of the capabilities attributes that manages lists of names 925 * (CA_SUNW_PLAT and CA_SUNW_MACH). Values are symbolic names that correspond 926 * to the capabilities mask in question. 927 * 928 * entry: 929 * mf - Mapfile descriptor 930 * eq_tok - One of TK_EQUAL, TK_PLUSEQ, TK_MINUSEQ, representing 931 * the operation to carry out. 932 * caplist - Caplist from output descriptor for capability being processed. 933 * type - Capability type (CA_SUNW_*) 934 * 935 * exit: 936 * Returns TK_SEMICOLON or TK_RIGHTBKT for success, and TK_ERROR otherwise. 937 */ 938 static Token 939 parse_cap_list(Mapfile *mf, Token eq_tok, Caplist *caplist, 940 Word type) 941 { 942 int done, found; 943 Token tok; 944 ld_map_tkval_t tkv; 945 Conv_inv_buf_t inv_buf; 946 APlist *strs = NULL; 947 Aliste idx; 948 const char *str; 949 950 for (done = 0, found = 0; done == 0; found = 0) { 951 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 952 case TK_ERROR: 953 return (TK_ERROR); 954 955 case TK_STRING: 956 /* 957 * The name is in tkv.tkv_str. Save this string for 958 * set_capstr() processing, but remove any duplicates. 959 */ 960 for (APLIST_TRAVERSE(strs, idx, str)) { 961 if (strcmp(str, tkv.tkv_str) == 0) { 962 found++; 963 break; 964 } 965 } 966 if ((found == 0) && (aplist_append(&strs, tkv.tkv_str, 967 AL_CNT_CAP_NAMES) == NULL)) 968 return (TK_ERROR); 969 break; 970 971 case TK_SEMICOLON: 972 case TK_RIGHTBKT: 973 done = 1; 974 break; 975 976 default: 977 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_CAPNAME), 978 ld_map_tokenstr(tok, &tkv, &inv_buf)); 979 return (TK_ERROR); 980 } 981 } 982 983 if (!set_capstr(mf, caplist, eq_tok, type, strs)) 984 return (TK_ERROR); 985 return (tok); 986 } 987 988 /* 989 * CAPABILITY [capid] { HW = hwcap_flags... 990 * -------------------------^ 991 */ 992 /* ARGSUSED 2 */ 993 static Token 994 at_cap_hw(Mapfile *mf, Token eq_tok, void *uvalue) 995 { 996 int done; 997 Token tok; 998 ld_map_tkval_t tkv; 999 Conv_inv_buf_t inv_buf; 1000 Word hw1 = 0, hw2 = 0; 1001 uint64_t v; 1002 1003 for (done = 0; done == 0; ) { 1004 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 1005 case TK_ERROR: 1006 return (TK_ERROR); 1007 1008 case TK_STRING: 1009 if ((v = elfcap_hw1_from_str(ELFCAP_STYLE, 1010 tkv.tkv_str, ld_targ.t_m.m_mach)) != 0) { 1011 hw1 |= v; 1012 break; 1013 } 1014 if ((v = elfcap_hw2_from_str(ELFCAP_STYLE, 1015 tkv.tkv_str, ld_targ.t_m.m_mach)) != 0) { 1016 hw2 |= v; 1017 break; 1018 } 1019 goto bad_flag; 1020 1021 case TK_SEMICOLON: 1022 case TK_RIGHTBKT: 1023 done = 1; 1024 break; 1025 1026 default: 1027 bad_flag: 1028 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_CAPHW), 1029 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1030 return (TK_ERROR); 1031 } 1032 } 1033 1034 if (!set_capmask(mf, &mf->mf_ofl->ofl_ocapset.oc_hw_1, eq_tok, 1035 CA_SUNW_HW_1, hw1, TRUE)) 1036 return (TK_ERROR); 1037 if (!set_capmask(mf, &mf->mf_ofl->ofl_ocapset.oc_hw_2, eq_tok, 1038 CA_SUNW_HW_2, hw2, FALSE)) 1039 return (TK_ERROR); 1040 return (tok); 1041 } 1042 1043 /* 1044 * CAPABILITY [capid] { HW_1 = value ; 1045 * ---------------------------^ 1046 */ 1047 /* ARGSUSED 2 */ 1048 static Token 1049 at_cap_hw_1(Mapfile *mf, Token eq_tok, void *uvalue) 1050 { 1051 return (parse_cap_mask(mf, eq_tok, &mf->mf_ofl->ofl_ocapset.oc_hw_1, 1052 CA_SUNW_HW_1, elfcap_hw1_from_str)); 1053 } 1054 1055 /* 1056 * CAPABILITY [capid] { HW_2 = value ; 1057 * ---------------------------^ 1058 */ 1059 /* ARGSUSED 2 */ 1060 static Token 1061 at_cap_hw_2(Mapfile *mf, Token eq_tok, void *uvalue) 1062 { 1063 return (parse_cap_mask(mf, eq_tok, &mf->mf_ofl->ofl_ocapset.oc_hw_2, 1064 CA_SUNW_HW_2, elfcap_hw2_from_str)); 1065 } 1066 1067 /* 1068 * CAPABILITY [capid] { SF = sfcap_flags... 1069 * -------------------------^ 1070 */ 1071 /* ARGSUSED 2 */ 1072 static Token 1073 at_cap_sf(Mapfile *mf, Token eq_tok, void *uvalue) 1074 { 1075 int done; 1076 Token tok; 1077 ld_map_tkval_t tkv; 1078 Conv_inv_buf_t inv_buf; 1079 Word sf1 = 0; 1080 uint64_t v; 1081 1082 for (done = 0; done == 0; ) { 1083 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 1084 case TK_ERROR: 1085 return (TK_ERROR); 1086 1087 case TK_STRING: 1088 if ((v = elfcap_sf1_from_str(ELFCAP_STYLE, 1089 tkv.tkv_str, ld_targ.t_m.m_mach)) != 0) { 1090 sf1 |= v; 1091 break; 1092 } 1093 goto bad_flag; 1094 1095 case TK_SEMICOLON: 1096 case TK_RIGHTBKT: 1097 done = 1; 1098 break; 1099 1100 default: 1101 bad_flag: 1102 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_CAPSF), 1103 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1104 return (TK_ERROR); 1105 } 1106 } 1107 1108 if (!set_capmask(mf, &mf->mf_ofl->ofl_ocapset.oc_sf_1, eq_tok, 1109 CA_SUNW_SF_1, sf1, TRUE)) 1110 return (TK_ERROR); 1111 1112 return (tok); 1113 } 1114 1115 /* 1116 * CAPABILITY [capid] { SF_1 = value ; 1117 * ---------------------------^ 1118 */ 1119 /* ARGSUSED 2 */ 1120 static Token 1121 at_cap_sf_1(Mapfile *mf, Token eq_tok, void *uvalue) 1122 { 1123 return (parse_cap_mask(mf, eq_tok, &mf->mf_ofl->ofl_ocapset.oc_sf_1, 1124 CA_SUNW_SF_1, elfcap_sf1_from_str)); 1125 } 1126 1127 /* 1128 * CAPABILITY [capid] { MACHINE = value ; 1129 * ------------------------------^ 1130 */ 1131 /* ARGSUSED 2 */ 1132 static Token 1133 at_cap_mach(Mapfile *mf, Token eq_tok, void *uvalue) 1134 { 1135 return (parse_cap_list(mf, eq_tok, &mf->mf_ofl->ofl_ocapset.oc_mach, 1136 CA_SUNW_MACH)); 1137 } 1138 1139 /* 1140 * CAPABILITY [capid] { PLATFORM = value ; 1141 * -------------------------------^ 1142 */ 1143 /* ARGSUSED 2 */ 1144 static Token 1145 at_cap_plat(Mapfile *mf, Token eq_tok, void *uvalue) 1146 { 1147 return (parse_cap_list(mf, eq_tok, &mf->mf_ofl->ofl_ocapset.oc_plat, 1148 CA_SUNW_PLAT)); 1149 } 1150 1151 /* 1152 * Top Level Directive: 1153 * 1154 * CAPABILITY [capid] { ... 1155 * ----------^ 1156 */ 1157 static Token 1158 dir_capability(Mapfile *mf) 1159 { 1160 /* CAPABILITY attributes */ 1161 static attr_t attr_list[] = { 1162 { MSG_ORIG(MSG_MAPKW_HW), at_cap_hw, ATTR_FMT_EQ_ALL }, 1163 { MSG_ORIG(MSG_MAPKW_HW_1), at_cap_hw_1, ATTR_FMT_EQ_ALL }, 1164 { MSG_ORIG(MSG_MAPKW_HW_2), at_cap_hw_2, ATTR_FMT_EQ_ALL }, 1165 1166 { MSG_ORIG(MSG_MAPKW_MACHINE), at_cap_mach, ATTR_FMT_EQ_ALL }, 1167 { MSG_ORIG(MSG_MAPKW_PLATFORM), at_cap_plat, ATTR_FMT_EQ_ALL }, 1168 1169 { MSG_ORIG(MSG_MAPKW_SF), at_cap_sf, ATTR_FMT_EQ_ALL }, 1170 { MSG_ORIG(MSG_MAPKW_SF_1), at_cap_sf_1, ATTR_FMT_EQ_ALL }, 1171 1172 /* List must be null terminated */ 1173 { 0 } 1174 }; 1175 1176 /* 1177 * Size of buffer needed to format the names in attr_list[]. Must 1178 * be kept in sync with attr_list. 1179 */ 1180 static size_t attr_list_bufsize = 1181 KW_NAME_SIZE(MSG_MAPKW_HW) + 1182 KW_NAME_SIZE(MSG_MAPKW_HW_1) + 1183 KW_NAME_SIZE(MSG_MAPKW_HW_2) + 1184 KW_NAME_SIZE(MSG_MAPKW_MACHINE) + 1185 KW_NAME_SIZE(MSG_MAPKW_PLATFORM) + 1186 KW_NAME_SIZE(MSG_MAPKW_SF) + 1187 KW_NAME_SIZE(MSG_MAPKW_SF_1); 1188 1189 Capstr *capstr; 1190 Token tok; 1191 ld_map_tkval_t tkv; 1192 Conv_inv_buf_t inv_buf; 1193 1194 /* 1195 * The first token can be one of: 1196 * - An opening '{' 1197 * - A name, followed by a '{', or a ';'. 1198 * Read this initial sequence. 1199 */ 1200 1201 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 1202 case TK_ERROR: 1203 return (TK_ERROR); 1204 1205 case TK_STRING: 1206 capstr = &mf->mf_ofl->ofl_ocapset.oc_id; 1207 1208 /* 1209 * The ID name is in tkv.tkv_str. Save this name in the output 1210 * capabilities structure. Note, should multiple ID entries 1211 * be encounterd, the last entry wins. 1212 */ 1213 DBG_CALL(Dbg_cap_id(mf->mf_ofl->ofl_lml, mf->mf_lineno, 1214 capstr->cs_str, tkv.tkv_str)); 1215 1216 capstr->cs_str = tkv.tkv_str; 1217 mf->mf_ofl->ofl_ocapset.oc_flags |= FLG_OCS_USRDEFID; 1218 1219 /* 1220 * The name can be followed by an opening '{', or a 1221 * terminating ';' 1222 */ 1223 switch (tok = gettoken_optattr(mf, capstr->cs_str)) { 1224 case TK_SEMICOLON: 1225 return (TK_SEMICOLON); 1226 case TK_LEFTBKT: 1227 break; 1228 default: 1229 return (TK_ERROR); 1230 } 1231 break; 1232 1233 case TK_LEFTBKT: 1234 /* Directive has no capid, but does supply attributes */ 1235 break; 1236 1237 default: 1238 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_CAPID), 1239 MSG_ORIG(MSG_MAPKW_CAPABILITY), 1240 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1241 return (TK_ERROR); 1242 } 1243 1244 /* Parse the attributes */ 1245 if (parse_attributes(mf, MSG_ORIG(MSG_MAPKW_CAPABILITY), 1246 attr_list, attr_list_bufsize, NULL) == TK_ERROR) 1247 return (TK_ERROR); 1248 1249 /* Terminating ';' */ 1250 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_CAPABILITY))); 1251 } 1252 1253 /* 1254 * at_dv_allow(): Value for ALLOW= is not a version string 1255 */ 1256 static void 1257 gts_efunc_at_dv_allow(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1258 { 1259 Conv_inv_buf_t inv_buf; 1260 1261 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_VERSION), 1262 MSG_ORIG(MSG_MAPKW_ALLOW), ld_map_tokenstr(tok, tkv, &inv_buf)); 1263 } 1264 1265 /* 1266 * DEPEND_VERSIONS object_name { ALLOW = version 1267 * -------------------------------------^ 1268 */ 1269 /* ARGSUSED 1 */ 1270 static Token 1271 at_dv_allow(Mapfile *mf, Token eq_tok, void *uvalue) 1272 { 1273 ld_map_tkval_t tkv; 1274 1275 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_dv_allow) == TK_ERROR) 1276 return (TK_ERROR); 1277 1278 /* Enter the version. uvalue points at the Sdf_desc descriptor */ 1279 if (!ld_map_dv_entry(mf, uvalue, FALSE, tkv.tkv_str)) 1280 return (TK_ERROR); 1281 1282 /* terminator */ 1283 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_ALLOW))); 1284 } 1285 1286 /* 1287 * at_dv_allow(): Value for REQUIRE= is not a version string 1288 */ 1289 static void 1290 gts_efunc_at_dv_require(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1291 { 1292 Conv_inv_buf_t inv_buf; 1293 1294 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_VERSION), 1295 MSG_ORIG(MSG_MAPKW_REQUIRE), ld_map_tokenstr(tok, tkv, &inv_buf)); 1296 } 1297 1298 /* 1299 * DEPEND_VERSIONS object_name { REQURE = version 1300 * --------------------------------------^ 1301 */ 1302 /* ARGSUSED 1 */ 1303 static Token 1304 at_dv_require(Mapfile *mf, Token eq_tok, void *uvalue) 1305 { 1306 ld_map_tkval_t tkv; 1307 1308 /* version_name */ 1309 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_dv_require) == TK_ERROR) 1310 return (TK_ERROR); 1311 1312 /* Enter the version. uvalue points at the Sdf_desc descriptor */ 1313 if (!ld_map_dv_entry(mf, uvalue, TRUE, tkv.tkv_str)) 1314 return (TK_ERROR); 1315 1316 /* terminator */ 1317 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_REQUIRE))); 1318 } 1319 1320 /* 1321 * dir_depend_versions(): Expected object name is not present 1322 */ 1323 static void 1324 gts_efunc_dir_depend_versions(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1325 { 1326 Conv_inv_buf_t inv_buf; 1327 1328 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_OBJNAM), 1329 MSG_ORIG(MSG_MAPKW_DEPEND_VERSIONS), 1330 ld_map_tokenstr(tok, tkv, &inv_buf)); 1331 } 1332 1333 /* 1334 * Top Level Directive: 1335 * 1336 * DEPEND_VERSIONS object_name { ATTR = ... 1337 * ---------------^ 1338 */ 1339 static Token 1340 dir_depend_versions(Mapfile *mf) 1341 { 1342 /* DEPEND_VERSIONS attributes */ 1343 static attr_t attr_list[] = { 1344 { MSG_ORIG(MSG_MAPKW_ALLOW), at_dv_allow, ATTR_FMT_EQ }, 1345 { MSG_ORIG(MSG_MAPKW_REQUIRE), at_dv_require, ATTR_FMT_EQ }, 1346 1347 /* List must be null terminated */ 1348 { 0 } 1349 }; 1350 1351 /* 1352 * Size of buffer needed to format the names in attr_list[]. Must 1353 * be kept in sync with attr_list. 1354 */ 1355 static size_t attr_list_bufsize = 1356 KW_NAME_SIZE(MSG_MAPKW_ALLOW) + 1357 KW_NAME_SIZE(MSG_MAPKW_REQUIRE); 1358 1359 ld_map_tkval_t tkv; 1360 Sdf_desc *sdf; 1361 1362 /* object_name */ 1363 if (gettoken_str(mf, 0, &tkv, gts_efunc_dir_depend_versions) == 1364 TK_ERROR) 1365 return (TK_ERROR); 1366 1367 /* Get descriptor for dependency */ 1368 if ((sdf = ld_map_dv(mf, tkv.tkv_str)) == NULL) 1369 return (TK_ERROR); 1370 1371 /* Opening '{' token */ 1372 if (gettoken_leftbkt(mf, tkv.tkv_str) == TK_ERROR) 1373 return (TK_ERROR); 1374 1375 /* Parse the attributes */ 1376 if (parse_attributes(mf, MSG_ORIG(MSG_MAPKW_DEPEND_VERSIONS), 1377 attr_list, attr_list_bufsize, sdf) == TK_ERROR) 1378 return (TK_ERROR); 1379 1380 /* Terminating ';' */ 1381 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_DEPEND_VERSIONS))); 1382 } 1383 1384 /* 1385 * Top Level Directive: 1386 * 1387 * HDR_NOALLOC ; 1388 * -----------^ 1389 */ 1390 static Token 1391 dir_hdr_noalloc(Mapfile *mf) 1392 { 1393 mf->mf_ofl->ofl_dtflags_1 |= DF_1_NOHDR; 1394 DBG_CALL(Dbg_map_hdr_noalloc(mf->mf_ofl->ofl_lml, mf->mf_lineno)); 1395 1396 /* ';' terminator token */ 1397 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_HDR_NOALLOC))); 1398 } 1399 1400 /* 1401 * Top Level Directive: 1402 * 1403 * PHDR_ADD_NULL = cnt ; 1404 * -------------^ 1405 */ 1406 static Token 1407 dir_phdr_add_null(Mapfile *mf) 1408 { 1409 Sg_desc *sgp; 1410 ld_map_tkval_t tkv; /* Value of token */ 1411 1412 /* '=' token */ 1413 if (gettoken_eq(mf, ATTR_FMT_EQ, 1414 MSG_ORIG(MSG_MAPKW_PHDR_ADD_NULL)) == TK_ERROR) 1415 return (TK_ERROR); 1416 1417 /* integer token */ 1418 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_PHDR_ADD_NULL), &tkv) == 1419 TK_ERROR) 1420 return (TK_ERROR); 1421 1422 while (tkv.tkv_int.tkvi_value-- > 0) { 1423 if ((sgp = ld_map_seg_alloc(NULL, PT_NULL, 1424 FLG_SG_P_TYPE | FLG_SG_EMPTY)) == NULL) 1425 return (TK_ERROR); 1426 if (ld_map_seg_insert(mf, DBG_STATE_NEW, sgp, 0) == 1427 SEG_INS_FAIL) 1428 return (TK_ERROR); 1429 } 1430 1431 /* ';' terminator token */ 1432 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_PHDR_ADD_NULL))); 1433 } 1434 1435 /* 1436 * segment_directive segment_name { ALIGN = value 1437 * ----------------------------------------^ 1438 */ 1439 /* ARGSUSED 1 */ 1440 static Token 1441 at_seg_align(Mapfile *mf, Token eq_tok, void *uvalue) 1442 { 1443 Sg_desc *sgp = uvalue; 1444 ld_map_tkval_t tkv; 1445 1446 /* value */ 1447 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_ALIGN), &tkv) == TK_ERROR) 1448 return (TK_ERROR); 1449 1450 sgp->sg_phdr.p_align = tkv.tkv_int.tkvi_value; 1451 sgp->sg_flags |= FLG_SG_P_ALIGN; 1452 1453 /* terminator */ 1454 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_ALIGN))); 1455 } 1456 1457 /* 1458 * at_seg_assign_file_basename(): Value for FILE_BASENAME= is not a file name 1459 */ 1460 static void 1461 gts_efunc_at_seg_assign_file_basename(Mapfile *mf, Token tok, 1462 ld_map_tkval_t *tkv) 1463 { 1464 Conv_inv_buf_t inv_buf; 1465 1466 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_FILNAM), 1467 MSG_ORIG(MSG_MAPKW_FILE_BASENAME), 1468 ld_map_tokenstr(tok, tkv, &inv_buf)); 1469 } 1470 1471 /* 1472 * segment_directive segment_name { ASSIGN { FILE_BASENAME = file_name 1473 * ---------------------------------------------------------^ 1474 */ 1475 /* ARGSUSED 1 */ 1476 static Token 1477 at_seg_assign_file_basename(Mapfile *mf, Token eq_tok, void *uvalue) 1478 { 1479 Ent_desc *enp = uvalue; 1480 ld_map_tkval_t tkv; 1481 1482 /* file_name */ 1483 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_seg_assign_file_basename) == 1484 TK_ERROR) 1485 return (TK_ERROR); 1486 1487 if (!ld_map_seg_ent_files(mf, enp, TYP_ECF_BASENAME, tkv.tkv_str)) 1488 return (TK_ERROR); 1489 1490 /* terminator */ 1491 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_FILE_BASENAME))); 1492 } 1493 1494 /* 1495 * at_seg_assign_file_objname(): Value for FILE_OBJNAME= is not an object name 1496 */ 1497 static void 1498 gts_efunc_at_seg_assign_file_objname(Mapfile *mf, Token tok, 1499 ld_map_tkval_t *tkv) 1500 { 1501 Conv_inv_buf_t inv_buf; 1502 1503 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_OBJNAM), 1504 MSG_ORIG(MSG_MAPKW_FILE_OBJNAME), 1505 ld_map_tokenstr(tok, tkv, &inv_buf)); 1506 } 1507 1508 /* 1509 * segment_directive segment_name { ASSIGN { FILE_OBJNAME = name 1510 * --------------------------------------------------------^ 1511 */ 1512 /* ARGSUSED 1 */ 1513 static Token 1514 at_seg_assign_file_objname(Mapfile *mf, Token eq_tok, void *uvalue) 1515 { 1516 Ent_desc *enp = uvalue; 1517 ld_map_tkval_t tkv; 1518 1519 /* file_objname */ 1520 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_seg_assign_file_objname) == 1521 TK_ERROR) 1522 return (TK_ERROR); 1523 1524 if (!ld_map_seg_ent_files(mf, enp, TYP_ECF_OBJNAME, tkv.tkv_str)) 1525 return (TK_ERROR); 1526 1527 /* terminator */ 1528 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_FILE_OBJNAME))); 1529 } 1530 1531 /* 1532 * at_seg_assign_file_path(): Value for FILE_PATH= is not a file path 1533 */ 1534 static void 1535 gts_efunc_at_seg_assign_file_path(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1536 { 1537 Conv_inv_buf_t inv_buf; 1538 1539 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_FILPATH), 1540 MSG_ORIG(MSG_MAPKW_FILE_PATH), 1541 ld_map_tokenstr(tok, tkv, &inv_buf)); 1542 } 1543 1544 /* 1545 * segment_directive segment_name { ASSIGN { FILE_PATH = file_path 1546 * -----------------------------------------------------^ 1547 */ 1548 /* ARGSUSED 1 */ 1549 static Token 1550 at_seg_assign_file_path(Mapfile *mf, Token eq_tok, void *uvalue) 1551 { 1552 Ent_desc *enp = uvalue; 1553 ld_map_tkval_t tkv; 1554 1555 /* file_path */ 1556 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_seg_assign_file_path) == 1557 TK_ERROR) 1558 return (TK_ERROR); 1559 1560 if (!ld_map_seg_ent_files(mf, enp, TYP_ECF_PATH, tkv.tkv_str)) 1561 return (TK_ERROR); 1562 1563 /* terminator */ 1564 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_FILE_PATH))); 1565 } 1566 1567 /* 1568 * segment_directive segment_name { ASSIGN { FLAGS = ... ; 1569 * -------------------------------------------------^ 1570 */ 1571 /* ARGSUSED 1 */ 1572 static Token 1573 at_seg_assign_flags(Mapfile *mf, Token eq_tok, void *uvalue) 1574 { 1575 typedef struct { 1576 const char *name; 1577 Word value; 1578 } secflag_t; 1579 static secflag_t flag_list[] = { 1580 { MSG_ORIG(MSG_MAPKW_ALLOC), SHF_ALLOC }, 1581 { MSG_ORIG(MSG_MAPKW_EXECUTE), SHF_EXECINSTR }, 1582 { MSG_ORIG(MSG_MAPKW_WRITE), SHF_WRITE }, 1583 { MSG_ORIG(MSG_MAPKW_AMD64_LARGE), SHF_AMD64_LARGE }, 1584 1585 /* List must be null terminated */ 1586 { 0 }, 1587 }; 1588 1589 /* 1590 * Size of buffer needed to format the names in flag_list[]. Must 1591 * be kept in sync with flag_list. 1592 */ 1593 static size_t flag_list_bufsize = 1594 KW_NAME_SIZE(MSG_MAPKW_ALLOC) + 1595 KW_NAME_SIZE(MSG_MAPKW_EXECUTE) + 1596 KW_NAME_SIZE(MSG_MAPKW_WRITE) + 1597 KW_NAME_SIZE(MSG_MAPKW_AMD64_LARGE); 1598 1599 Ent_desc *enp = uvalue; 1600 int bcnt = 0, cnt = 0; 1601 secflag_t *flag; 1602 int done; 1603 Token tok; 1604 ld_map_tkval_t tkv; 1605 Conv_inv_buf_t inv_buf; 1606 1607 /* Read and process tokens until the closing terminator is seen */ 1608 for (done = 0; done == 0; ) { 1609 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 1610 case TK_ERROR: 1611 return (TK_ERROR); 1612 1613 case TK_BANG: 1614 /* Ensure ! only specified once per flag */ 1615 if (bcnt != 0) { 1616 mf_fatal0(mf, MSG_INTL(MSG_MAP_SFLG_ONEBANG)); 1617 return (TK_ERROR); 1618 } 1619 bcnt++; 1620 break; 1621 1622 case TK_STRING: 1623 flag = ld_map_kwfind(tkv.tkv_str, flag_list, 1624 SGSOFFSETOF(secflag_t, name), sizeof (flag[0])); 1625 if (flag == NULL) 1626 goto bad_flag; 1627 cnt++; 1628 enp->ec_attrmask |= flag->value; 1629 if (bcnt == 0) 1630 enp->ec_attrbits |= flag->value; 1631 bcnt = 0; 1632 break; 1633 1634 case TK_RIGHTBKT: 1635 case TK_SEMICOLON: 1636 done = 1; 1637 break; 1638 1639 default: 1640 bad_flag: 1641 { 1642 char buf[VLA_SIZE(flag_list_bufsize)]; 1643 1644 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SECFLAG), 1645 ld_map_kwnames(flag_list, 1646 SGSOFFSETOF(secflag_t, name), 1647 sizeof (flag[0]), buf, flag_list_bufsize), 1648 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1649 } 1650 return (TK_ERROR); 1651 } 1652 } 1653 1654 /* 1655 * Ensure that a trailing '!' was not left at the end of the line 1656 * without a corresponding flag to apply it to. 1657 */ 1658 if (bcnt != 0) { 1659 mf_fatal0(mf, MSG_INTL(MSG_MAP_SFLG_EXBANG)); 1660 return (TK_ERROR); 1661 } 1662 1663 /* Make sure there was at least one flag */ 1664 if (cnt == 0) { 1665 mf_fatal(mf, MSG_INTL(MSG_MAP_NOVALUES), 1666 MSG_ORIG(MSG_MAPKW_FLAGS)); 1667 return (TK_ERROR); 1668 } 1669 1670 return (tok); /* Either TK_SEMICOLON or TK_RIGHTBKT */ 1671 } 1672 1673 /* 1674 * at_seg_assign_is_name(): Value for IS_NAME= is not a section name 1675 */ 1676 static void 1677 gts_efunc_at_seg_assign_is_name(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1678 { 1679 Conv_inv_buf_t inv_buf; 1680 1681 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SECNAM), 1682 MSG_ORIG(MSG_MAPKW_IS_NAME), ld_map_tokenstr(tok, tkv, &inv_buf)); 1683 } 1684 1685 /* 1686 * segment_directive segment_name { ASSIGN { IS_NAME = section_name ; 1687 * ---------------------------------------------------^ 1688 */ 1689 /* ARGSUSED 1 */ 1690 static Token 1691 at_seg_assign_is_name(Mapfile *mf, Token eq_tok, void *uvalue) 1692 { 1693 Ent_desc *enp = uvalue; 1694 ld_map_tkval_t tkv; 1695 1696 /* section_name */ 1697 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_seg_assign_is_name) == 1698 TK_ERROR) 1699 return (TK_ERROR); 1700 enp->ec_is_name = tkv.tkv_str; 1701 1702 /* terminator */ 1703 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_IS_NAME))); 1704 } 1705 1706 /* 1707 * at_seg_assign_type(): Value for TYPE= is not a section type 1708 */ 1709 static void 1710 gts_efunc_at_seg_assign_type(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 1711 { 1712 Conv_inv_buf_t inv_buf; 1713 1714 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SHTYPE), 1715 ld_map_tokenstr(tok, tkv, &inv_buf)); 1716 } 1717 1718 /* 1719 * segment_directive segment_name { ASSIGN { TYPE = section_type ; 1720 * ------------------------------------------------^ 1721 */ 1722 /* ARGSUSED 1 */ 1723 static Token 1724 at_seg_assign_type(Mapfile *mf, Token eq_tok, void *uvalue) 1725 { 1726 Ent_desc *enp = uvalue; 1727 ld_map_tkval_t tkv; 1728 conv_strtol_uvalue_t conv_uvalue; 1729 1730 /* section type */ 1731 if (gettoken_str(mf, TK_F_KEYWORD, &tkv, 1732 gts_efunc_at_seg_assign_type) == TK_ERROR) 1733 return (TK_ERROR); 1734 1735 /* 1736 * Use the libconv iteration facility to map the given name to 1737 * its value. This allows us to keep up with any new sections 1738 * without having to change this code. 1739 */ 1740 if (conv_iter_strtol_init(tkv.tkv_str, &conv_uvalue) != 0) { 1741 conv_iter_ret_t status; 1742 1743 /* Look at the canonical form */ 1744 status = conv_iter_sec_type(CONV_OSABI_ALL, CONV_MACH_ALL, 1745 CONV_FMT_ALT_CF, conv_iter_strtol, &conv_uvalue); 1746 1747 /* Failing that, look at the normal form */ 1748 if (status != CONV_ITER_DONE) 1749 (void) conv_iter_sec_type(CONV_OSABI_ALL, 1750 CONV_MACH_ALL, CONV_FMT_ALT_NF, conv_iter_strtol, 1751 &conv_uvalue); 1752 1753 /* If we didn't match anything report error */ 1754 if (!conv_uvalue.csl_found) { 1755 gts_efunc_at_seg_assign_type(mf, TK_STRING, &tkv); 1756 return (TK_ERROR); 1757 } 1758 } 1759 1760 enp->ec_type = conv_uvalue.csl_value; 1761 1762 /* terminator */ 1763 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_TYPE))); 1764 } 1765 1766 /* 1767 * segment_directive segment_name { ASSIGN { ... 1768 * -----------------------------------------^ 1769 */ 1770 /* ARGSUSED 1 */ 1771 static Token 1772 at_seg_assign(Mapfile *mf, Token eq_tok, void *uvalue) 1773 { 1774 /* segment_directive ASSIGN sub-attributes */ 1775 static attr_t attr_list[] = { 1776 { MSG_ORIG(MSG_MAPKW_FILE_BASENAME), 1777 at_seg_assign_file_basename, ATTR_FMT_EQ }, 1778 { MSG_ORIG(MSG_MAPKW_FILE_OBJNAME), 1779 at_seg_assign_file_objname, ATTR_FMT_EQ }, 1780 { MSG_ORIG(MSG_MAPKW_FILE_PATH), 1781 at_seg_assign_file_path, ATTR_FMT_EQ }, 1782 { MSG_ORIG(MSG_MAPKW_FLAGS), 1783 at_seg_assign_flags, ATTR_FMT_EQ_ALL }, 1784 { MSG_ORIG(MSG_MAPKW_IS_NAME), 1785 at_seg_assign_is_name, ATTR_FMT_EQ }, 1786 { MSG_ORIG(MSG_MAPKW_TYPE), 1787 at_seg_assign_type, ATTR_FMT_EQ }, 1788 1789 /* List must be null terminated */ 1790 { 0 } 1791 }; 1792 1793 /* 1794 * Size of buffer needed to format the names in attr_list[]. Must 1795 * be kept in sync with attr_list. 1796 */ 1797 static size_t attr_list_bufsize = 1798 KW_NAME_SIZE(MSG_MAPKW_FILE_BASENAME) + 1799 KW_NAME_SIZE(MSG_MAPKW_FILE_PATH) + 1800 KW_NAME_SIZE(MSG_MAPKW_FLAGS) + 1801 KW_NAME_SIZE(MSG_MAPKW_FILE_OBJNAME) + 1802 KW_NAME_SIZE(MSG_MAPKW_IS_NAME) + 1803 KW_NAME_SIZE(MSG_MAPKW_TYPE); 1804 1805 Sg_desc *sgp = uvalue; 1806 Token tok; 1807 ld_map_tkval_t tkv; 1808 Conv_inv_buf_t inv_buf; 1809 const char *name = NULL; 1810 Ent_desc *enp; 1811 1812 /* 1813 * ASSIGN takes an optional name, plus attributes are optional, 1814 * so expect a name, an opening '{', or a ';'. 1815 */ 1816 tok = ld_map_gettoken(mf, 0, &tkv); 1817 switch (tok) { 1818 case TK_ERROR: 1819 return (TK_ERROR); 1820 1821 case TK_STRING: 1822 name = tkv.tkv_str; 1823 tok = ld_map_gettoken(mf, 0, &tkv); 1824 break; 1825 } 1826 1827 /* Add a new entrance criteria descriptor to the segment */ 1828 if ((enp = ld_map_seg_ent_add(mf, sgp, name)) == NULL) 1829 return (TK_ERROR); 1830 1831 /* Having handled the name, expect either '{' or ';' */ 1832 switch (tok) { 1833 default: 1834 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEMLBKT), 1835 MSG_ORIG(MSG_MAPKW_ASSIGN_SECTION), 1836 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1837 return (TK_ERROR); 1838 case TK_ERROR: 1839 return (TK_ERROR); 1840 case TK_SEMICOLON: 1841 case TK_RIGHTBKT: 1842 /* No attributes: It will match anything */ 1843 enp->ec_flags |= FLG_EC_CATCHALL; 1844 break; 1845 case TK_LEFTBKT: 1846 /* Parse the attributes */ 1847 if (parse_attributes(mf, MSG_ORIG(MSG_MAPKW_ASSIGN_SECTION), 1848 attr_list, attr_list_bufsize, enp) == TK_ERROR) 1849 return (TK_ERROR); 1850 1851 /* Terminating ';', or '}' which also terminates caller */ 1852 tok = gettoken_term(mf, MSG_ORIG(MSG_MAPKW_ASSIGN_SECTION)); 1853 if (tok == TK_ERROR) 1854 return (TK_ERROR); 1855 break; 1856 } 1857 1858 DBG_CALL(Dbg_map_ent(mf->mf_ofl->ofl_lml, enp, mf->mf_ofl, 1859 mf->mf_lineno)); 1860 return (tok); 1861 } 1862 1863 /* 1864 * segment_directive segment_name { DISABLE ; 1865 * ----------------------------------------^ 1866 */ 1867 /* ARGSUSED 1 */ 1868 static Token 1869 at_seg_disable(Mapfile *mf, Token eq_tok, void *uvalue) 1870 { 1871 Sg_desc *sgp = uvalue; 1872 1873 /* If the segment cannot be disabled, issue error */ 1874 if (sgp->sg_flags & FLG_SG_NODISABLE) { 1875 mf_fatal(mf, MSG_INTL(MSG_MAP_CNTDISSEG), sgp->sg_name); 1876 return (TK_ERROR); 1877 } 1878 1879 /* Disable the segment */ 1880 sgp->sg_flags |= FLG_SG_DISABLED; 1881 1882 /* terminator */ 1883 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_DISABLE))); 1884 } 1885 1886 /* 1887 * segment_directive segment_name { FLAGS eq-op ... 1888 * --------------------------------------------^ 1889 * 1890 * Note that this routine is also used for the STACK directive, 1891 * as STACK also manipulates a segment descriptor. 1892 * 1893 * STACK { FLAGS eq-op ... ; 1894 * -------------------^ 1895 */ 1896 /* ARGSUSED 2 */ 1897 static Token 1898 at_seg_flags(Mapfile *mf, Token eq_tok, void *uvalue) 1899 { 1900 Sg_desc *sgp = uvalue; 1901 Token tok; 1902 Xword flags; 1903 1904 tok = parse_segment_flags(mf, &flags); 1905 if (tok == TK_ERROR) 1906 return (TK_ERROR); 1907 1908 setflags_eq(&sgp->sg_phdr.p_flags, eq_tok, flags); 1909 sgp->sg_flags |= FLG_SG_P_FLAGS; 1910 1911 return (tok); 1912 } 1913 1914 /* 1915 * segment_directive segment_name { IS_ORDER eq_op value 1916 * -----------------------------------------------^ 1917 */ 1918 /* ARGSUSED 2 */ 1919 static Token 1920 at_seg_is_order(Mapfile *mf, Token eq_tok, void *uvalue) 1921 { 1922 Sg_desc *sgp = uvalue; 1923 Token tok; 1924 ld_map_tkval_t tkv; 1925 Conv_inv_buf_t inv_buf; 1926 int done; 1927 Aliste idx; 1928 Ent_desc *enp, *enp2; 1929 1930 /* 1931 * The '=' form of assignment resets the list. The list contains 1932 * pointers to our mapfile text, so we do not have to free anything. 1933 */ 1934 if (eq_tok == TK_EQUAL) 1935 aplist_reset(sgp->sg_is_order); 1936 1937 /* 1938 * One or more ASSIGN names, terminated by a semicolon. 1939 */ 1940 for (done = 0; done == 0; ) { 1941 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 1942 case TK_ERROR: 1943 return (TK_ERROR); 1944 1945 case TK_STRING: 1946 /* 1947 * The referenced entrance criteria must have 1948 * already been defined. 1949 */ 1950 enp = ld_ent_lookup(mf->mf_ofl, tkv.tkv_str, NULL); 1951 if (enp == NULL) { 1952 mf_fatal(mf, MSG_INTL(MSG_MAP_UNKENT), 1953 tkv.tkv_str); 1954 return (TK_ERROR); 1955 } 1956 1957 /* 1958 * Make sure it's not already on the list 1959 */ 1960 for (APLIST_TRAVERSE(sgp->sg_is_order, idx, enp2)) 1961 if (enp == enp2) { 1962 mf_fatal(mf, 1963 MSG_INTL(MSG_MAP_DUP_IS_ORD), 1964 tkv.tkv_str); 1965 return (TK_ERROR); 1966 } 1967 1968 /* Put it at the end of the order list */ 1969 if (aplist_append(&sgp->sg_is_order, enp, 1970 AL_CNT_SG_IS_ORDER) == NULL) 1971 return (TK_ERROR); 1972 break; 1973 1974 case TK_SEMICOLON: 1975 case TK_RIGHTBKT: 1976 done = 1; 1977 break; 1978 1979 default: 1980 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_ECNAM), 1981 ld_map_tokenstr(tok, &tkv, &inv_buf)); 1982 return (TK_ERROR); 1983 } 1984 } 1985 1986 return (tok); 1987 } 1988 1989 /* 1990 * segment_directive segment_name { MAX_SIZE = value 1991 * -------------------------------------------^ 1992 */ 1993 /* ARGSUSED 1 */ 1994 static Token 1995 at_seg_max_size(Mapfile *mf, Token eq_tok, void *uvalue) 1996 { 1997 Sg_desc *sgp = uvalue; 1998 ld_map_tkval_t tkv; 1999 2000 /* value */ 2001 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_MAX_SIZE), &tkv) == TK_ERROR) 2002 return (TK_ERROR); 2003 2004 sgp->sg_length = tkv.tkv_int.tkvi_value; 2005 sgp->sg_flags |= FLG_SG_LENGTH; 2006 2007 /* terminator */ 2008 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_MAX_SIZE))); 2009 } 2010 2011 /* 2012 * segment_directive segment_name { NOHDR ; 2013 * --------------------------------------^ 2014 */ 2015 /* ARGSUSED 1 */ 2016 static Token 2017 at_seg_nohdr(Mapfile *mf, Token eq_tok, void *uvalue) 2018 { 2019 Sg_desc *sgp = uvalue; 2020 2021 /* 2022 * Set the nohdr flag on the segment. If this segment is the 2023 * first loadable segment, the ELF and program headers will 2024 * not be included. 2025 * 2026 * The HDR_NOALLOC top level directive is preferred. This feature 2027 * exists to give 1:1 feature parity with version 1 mapfiles that 2028 * use the ?N segment flag and expect it to only take effect 2029 * if that segment ends up being first. 2030 */ 2031 sgp->sg_flags |= FLG_SG_NOHDR; 2032 2033 /* terminator */ 2034 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_NOHDR))); 2035 } 2036 2037 /* 2038 * segment_directive segment_name { OS_ORDER eq_op assign_name... 2039 * -----------------------------------------------^ 2040 */ 2041 /* ARGSUSED 2 */ 2042 static Token 2043 at_seg_os_order(Mapfile *mf, Token eq_tok, void *uvalue) 2044 { 2045 Sg_desc *sgp = uvalue; 2046 Token tok; 2047 ld_map_tkval_t tkv; 2048 Conv_inv_buf_t inv_buf; 2049 int done; 2050 2051 /* 2052 * The '=' form of assignment resets the list. The list contains 2053 * pointers to our mapfile text, so we do not have to free anything. 2054 */ 2055 if (eq_tok == TK_EQUAL) 2056 alist_reset(sgp->sg_os_order); 2057 2058 /* 2059 * One or more section names, terminated by a semicolon. 2060 */ 2061 for (done = 0; done == 0; ) { 2062 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 2063 case TK_ERROR: 2064 return (TK_ERROR); 2065 2066 case TK_STRING: 2067 if (!ld_map_seg_os_order_add(mf, sgp, tkv.tkv_str)) 2068 return (TK_ERROR); 2069 break; 2070 2071 case TK_SEMICOLON: 2072 case TK_RIGHTBKT: 2073 done = 1; 2074 break; 2075 2076 default: 2077 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SECNAM), 2078 ld_map_tokenstr(tok, &tkv, &inv_buf)); 2079 return (TK_ERROR); 2080 } 2081 } 2082 2083 return (tok); 2084 } 2085 2086 /* 2087 * segment_directive segment_name { PADDR = paddr 2088 * ----------------------------------------^ 2089 */ 2090 /* ARGSUSED 1 */ 2091 static Token 2092 at_seg_paddr(Mapfile *mf, Token eq_tok, void *uvalue) 2093 { 2094 Sg_desc *sgp = uvalue, *sgp2; 2095 Aliste idx; 2096 ld_map_tkval_t tkv; 2097 2098 /* 2099 * Ensure that the segment isn't in the segment order list. 2100 */ 2101 for (APLIST_TRAVERSE(mf->mf_ofl->ofl_segs_order, idx, sgp2)) 2102 if (sgp == sgp2) { 2103 mf_fatal(mf, 2104 MSG_INTL(MSG_MAP_CNTADDRORDER), sgp->sg_name); 2105 return (TK_ERROR); 2106 } 2107 2108 /* value */ 2109 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_PADDR), &tkv) == TK_ERROR) 2110 return (TK_ERROR); 2111 2112 sgp->sg_phdr.p_paddr = tkv.tkv_int.tkvi_value; 2113 sgp->sg_flags |= FLG_SG_P_PADDR; 2114 2115 /* terminator */ 2116 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_PADDR))); 2117 } 2118 2119 /* 2120 * segment_directive segment_name { ROUND = value 2121 * ----------------------------------------^ 2122 */ 2123 /* ARGSUSED 1 */ 2124 static Token 2125 at_seg_round(Mapfile *mf, Token eq_tok, void *uvalue) 2126 { 2127 Sg_desc *sgp = uvalue; 2128 ld_map_tkval_t tkv; 2129 2130 /* value */ 2131 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_ROUND), &tkv) == TK_ERROR) 2132 return (TK_ERROR); 2133 2134 sgp->sg_round = tkv.tkv_int.tkvi_value; 2135 sgp->sg_flags |= FLG_SG_ROUND; 2136 2137 /* terminator */ 2138 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_ROUND))); 2139 } 2140 2141 /* 2142 * segment_directive segment_name { SIZE_SYMBOL = symbol_name 2143 * ----------------------------------------------^ 2144 */ 2145 /* ARGSUSED 2 */ 2146 static Token 2147 at_seg_size_symbol(Mapfile *mf, Token eq_tok, void *uvalue) 2148 { 2149 Sg_desc *sgp = uvalue; 2150 Token tok; 2151 ld_map_tkval_t tkv; 2152 Conv_inv_buf_t inv_buf; 2153 int done, cnt = 0; 2154 2155 /* 2156 * One or more symbol names, terminated by a semicolon. 2157 */ 2158 for (done = 0; done == 0; ) { 2159 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 2160 case TK_ERROR: 2161 return (TK_ERROR); 2162 2163 case TK_STRING: 2164 if (!ld_map_seg_size_symbol(mf, sgp, eq_tok, 2165 tkv.tkv_str)) 2166 return (TK_ERROR); 2167 cnt++; 2168 2169 /* 2170 * If the operator is TK_EQUAL, turn it into 2171 * TK_PLUSEQ for any symbol names after the first. 2172 * These additional symbols are added, and are not 2173 * replacements for the first one. 2174 */ 2175 eq_tok = TK_PLUSEQ; 2176 break; 2177 2178 case TK_SEMICOLON: 2179 case TK_RIGHTBKT: 2180 done = 1; 2181 break; 2182 2183 default: 2184 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SYMNAM), 2185 MSG_ORIG(MSG_MAPKW_SIZE_SYMBOL), 2186 ld_map_tokenstr(tok, &tkv, &inv_buf)); 2187 return (TK_ERROR); 2188 } 2189 } 2190 2191 /* Make sure there was at least one name */ 2192 if (cnt == 0) { 2193 mf_fatal(mf, MSG_INTL(MSG_MAP_NOVALUES), 2194 MSG_ORIG(MSG_MAPKW_SIZE_SYMBOL)); 2195 return (TK_ERROR); 2196 } 2197 2198 return (tok); 2199 } 2200 2201 /* 2202 * segment_directive segment_name { VADDR = vaddr 2203 * ----------------------------------------^ 2204 */ 2205 /* ARGSUSED 1 */ 2206 static Token 2207 at_seg_vaddr(Mapfile *mf, Token eq_tok, void *uvalue) 2208 { 2209 Sg_desc *sgp = uvalue, *sgp2; 2210 Aliste idx; 2211 ld_map_tkval_t tkv; 2212 2213 /* 2214 * Ensure that the segment isn't in the segment order list. 2215 */ 2216 for (APLIST_TRAVERSE(mf->mf_ofl->ofl_segs_order, idx, sgp2)) 2217 if (sgp == sgp2) { 2218 mf_fatal(mf, 2219 MSG_INTL(MSG_MAP_CNTADDRORDER), sgp->sg_name); 2220 return (TK_ERROR); 2221 } 2222 2223 /* value */ 2224 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_VADDR), &tkv) == TK_ERROR) 2225 return (TK_ERROR); 2226 2227 sgp->sg_phdr.p_vaddr = tkv.tkv_int.tkvi_value; 2228 sgp->sg_flags |= FLG_SG_P_VADDR; 2229 2230 /* terminator */ 2231 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_VADDR))); 2232 } 2233 2234 /* 2235 * Top Level Directive: 2236 * 2237 * {LOAD|NOTE|NULL}_SEGMENT segment_name { ... 2238 * ------------------------^ 2239 * 2240 * Common implementation body for the family of segment directives. These 2241 * take the same syntax, and share a common subset of attributes. They differ 2242 * in the type of segments they handle and the specific attributes accepted. 2243 * 2244 * entry: 2245 * mf - Mapfile descriptor ({LOAD|NOTE|NULL}_SEGMENT) 2246 * dir_name - Name of directive. 2247 * seg_type - Type of segment (PT_LOAD, PT_NOTE, PT_NULL). 2248 * attr_list - NULL terminated attribute array 2249 * attr_list_bufsize - Size of required buffer to format all the 2250 * names in attr_list. 2251 * gts_efunc - Error function to pass to gettoken_str() when trying 2252 * to obtain a segment name token. 2253 */ 2254 static Token 2255 dir_segment_inner(Mapfile *mf, const char *dir_name, Word seg_type, 2256 attr_t *attr_list, size_t attr_list_bufsize, gts_efunc_t gts_efunc) 2257 { 2258 Token tok; 2259 ld_map_tkval_t tkv; 2260 Sg_desc *sgp; 2261 Boolean new_segment; 2262 Xword ndx; 2263 avl_index_t where; 2264 2265 /* segment_name */ 2266 if (gettoken_str(mf, 0, &tkv, gts_efunc) == TK_ERROR) 2267 return (TK_ERROR); 2268 sgp = ld_seg_lookup(mf->mf_ofl, tkv.tkv_str, &where); 2269 new_segment = (sgp == NULL); 2270 2271 if (new_segment) { 2272 /* Allocate a descriptor for new segment */ 2273 if ((sgp = ld_map_seg_alloc(tkv.tkv_str, seg_type, 2274 FLG_SG_P_TYPE)) == NULL) 2275 return (TK_ERROR); 2276 } else { 2277 /* Make sure it's the right type of segment */ 2278 if (sgp->sg_phdr.p_type != seg_type) { 2279 Conv_inv_buf_t inv_buf; 2280 2281 mf_fatal(mf, MSG_INTL(MSG_MAP_EXPSEGTYPE), 2282 conv_phdr_type(ELFOSABI_SOLARIS, ld_targ.t_m.m_mach, 2283 sgp->sg_phdr.p_type, CONV_FMT_ALT_CF, &inv_buf), 2284 dir_name, tkv.tkv_str); 2285 return (TK_ERROR); 2286 } 2287 2288 /* If it was disabled, being referenced enables it */ 2289 sgp->sg_flags &= ~FLG_SG_DISABLED; 2290 2291 if (DBG_ENABLED) { 2292 /* 2293 * Not a new segment, so show the initial value 2294 * before modifying it. 2295 */ 2296 ndx = ld_map_seg_index(mf, sgp); 2297 DBG_CALL(Dbg_map_seg(mf->mf_ofl, DBG_STATE_MOD_BEFORE, 2298 ndx, sgp, mf->mf_lineno)); 2299 } 2300 } 2301 2302 /* 2303 * Attributes are optional, so expect an opening '{', or a ';'. 2304 */ 2305 switch (tok = gettoken_optattr(mf, dir_name)) { 2306 default: 2307 tok = TK_ERROR; 2308 break; 2309 case TK_SEMICOLON: 2310 break; 2311 case TK_LEFTBKT: 2312 /* Parse the attributes */ 2313 if (parse_attributes(mf, dir_name, 2314 attr_list, attr_list_bufsize, sgp) == TK_ERROR) 2315 return (TK_ERROR); 2316 2317 /* Terminating ';' */ 2318 tok = gettoken_semicolon(mf, dir_name); 2319 if (tok == TK_ERROR) 2320 return (TK_ERROR); 2321 2322 break; 2323 } 2324 2325 /* 2326 * If this is a new segment, finish its initialization 2327 * and insert it into the segment list. 2328 */ 2329 if (new_segment) { 2330 if (ld_map_seg_insert(mf, DBG_STATE_NEW, sgp, where) == 2331 SEG_INS_FAIL) 2332 return (TK_ERROR); 2333 } else { 2334 /* Not new. Show what's changed */ 2335 DBG_CALL(Dbg_map_seg(mf->mf_ofl, DBG_STATE_MOD_AFTER, 2336 ndx, sgp, mf->mf_lineno)); 2337 } 2338 2339 return (tok); 2340 } 2341 2342 /* 2343 * dir_load_segment(): Expected loadable segment name is not present 2344 */ 2345 static void 2346 gts_efunc_dir_load_segment(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2347 { 2348 Conv_inv_buf_t inv_buf; 2349 2350 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEGNAM), 2351 MSG_ORIG(MSG_MAPKW_LOAD_SEGMENT), 2352 ld_map_tokenstr(tok, tkv, &inv_buf)); 2353 } 2354 2355 /* 2356 * Top Level Directive: 2357 * 2358 * LOAD_SEGMENT segment_name { ... 2359 * ------------^ 2360 */ 2361 static Token 2362 dir_load_segment(Mapfile *mf) 2363 { 2364 /* LOAD_SEGMENT attributes */ 2365 static attr_t attr_list[] = { 2366 { MSG_ORIG(MSG_MAPKW_ALIGN), at_seg_align, ATTR_FMT_EQ }, 2367 { MSG_ORIG(MSG_MAPKW_ASSIGN_SECTION), 2368 at_seg_assign, ATTR_FMT_NAME }, 2369 { MSG_ORIG(MSG_MAPKW_DISABLE), at_seg_disable, ATTR_FMT_NAME }, 2370 { MSG_ORIG(MSG_MAPKW_FLAGS), at_seg_flags, 2371 ATTR_FMT_EQ_ALL }, 2372 { MSG_ORIG(MSG_MAPKW_IS_ORDER), at_seg_is_order, 2373 ATTR_FMT_EQ_PEQ }, 2374 { MSG_ORIG(MSG_MAPKW_MAX_SIZE), at_seg_max_size, ATTR_FMT_EQ }, 2375 { MSG_ORIG(MSG_MAPKW_NOHDR), at_seg_nohdr, ATTR_FMT_NAME }, 2376 { MSG_ORIG(MSG_MAPKW_OS_ORDER), at_seg_os_order, 2377 ATTR_FMT_EQ_PEQ }, 2378 { MSG_ORIG(MSG_MAPKW_PADDR), at_seg_paddr, ATTR_FMT_EQ }, 2379 { MSG_ORIG(MSG_MAPKW_ROUND), at_seg_round, ATTR_FMT_EQ }, 2380 { MSG_ORIG(MSG_MAPKW_SIZE_SYMBOL), 2381 at_seg_size_symbol, ATTR_FMT_EQ_PEQ }, 2382 { MSG_ORIG(MSG_MAPKW_VADDR), at_seg_vaddr, ATTR_FMT_EQ }, 2383 2384 /* List must be null terminated */ 2385 { 0 } 2386 }; 2387 2388 /* 2389 * Size of buffer needed to format the names in attr_list[]. Must 2390 * be kept in sync with attr_list. 2391 */ 2392 static size_t attr_list_bufsize = 2393 KW_NAME_SIZE(MSG_MAPKW_ALIGN) + 2394 KW_NAME_SIZE(MSG_MAPKW_ASSIGN_SECTION) + 2395 KW_NAME_SIZE(MSG_MAPKW_DISABLE) + 2396 KW_NAME_SIZE(MSG_MAPKW_FLAGS) + 2397 KW_NAME_SIZE(MSG_MAPKW_IS_ORDER) + 2398 KW_NAME_SIZE(MSG_MAPKW_MAX_SIZE) + 2399 KW_NAME_SIZE(MSG_MAPKW_PADDR) + 2400 KW_NAME_SIZE(MSG_MAPKW_ROUND) + 2401 KW_NAME_SIZE(MSG_MAPKW_OS_ORDER) + 2402 KW_NAME_SIZE(MSG_MAPKW_SIZE_SYMBOL) + 2403 KW_NAME_SIZE(MSG_MAPKW_VADDR); 2404 2405 return (dir_segment_inner(mf, MSG_ORIG(MSG_MAPKW_LOAD_SEGMENT), 2406 PT_LOAD, attr_list, attr_list_bufsize, gts_efunc_dir_load_segment)); 2407 2408 } 2409 2410 /* 2411 * Common shared segment directive attributes 2412 */ 2413 static attr_t segment_core_attr_list[] = { 2414 { MSG_ORIG(MSG_MAPKW_ASSIGN_SECTION), at_seg_assign, ATTR_FMT_NAME }, 2415 { MSG_ORIG(MSG_MAPKW_DISABLE), at_seg_disable, ATTR_FMT_NAME }, 2416 { MSG_ORIG(MSG_MAPKW_IS_ORDER), at_seg_is_order, ATTR_FMT_EQ_PEQ }, 2417 { MSG_ORIG(MSG_MAPKW_OS_ORDER), at_seg_os_order, ATTR_FMT_EQ_PEQ }, 2418 2419 /* List must be null terminated */ 2420 { 0 } 2421 }; 2422 2423 /* 2424 * Size of buffer needed to format the names in segment_core_attr_list[]. 2425 * Must be kept in sync with segment_core_attr_list. 2426 */ 2427 static size_t segment_core_attr_list_bufsize = 2428 KW_NAME_SIZE(MSG_MAPKW_ASSIGN_SECTION) + 2429 KW_NAME_SIZE(MSG_MAPKW_DISABLE) + 2430 KW_NAME_SIZE(MSG_MAPKW_IS_ORDER) + 2431 KW_NAME_SIZE(MSG_MAPKW_OS_ORDER); 2432 2433 /* 2434 * dir_note_segment(): Expected note segment name is not present 2435 */ 2436 static void 2437 gts_efunc_dir_note_segment(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2438 { 2439 Conv_inv_buf_t inv_buf; 2440 2441 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEGNAM), 2442 MSG_ORIG(MSG_MAPKW_NOTE_SEGMENT), 2443 ld_map_tokenstr(tok, tkv, &inv_buf)); 2444 } 2445 2446 /* 2447 * Top Level Directive: 2448 * 2449 * NOTE_SEGMENT segment_name { ... 2450 * ------------^ 2451 */ 2452 static Token 2453 dir_note_segment(Mapfile *mf) 2454 { 2455 return (dir_segment_inner(mf, MSG_ORIG(MSG_MAPKW_NOTE_SEGMENT), 2456 PT_NOTE, segment_core_attr_list, segment_core_attr_list_bufsize, 2457 gts_efunc_dir_note_segment)); 2458 2459 } 2460 2461 /* 2462 * dir_null_segment(): Expected null segment name is not present 2463 */ 2464 static void 2465 gts_efunc_dir_null_segment(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2466 { 2467 Conv_inv_buf_t inv_buf; 2468 2469 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEGNAM), 2470 MSG_ORIG(MSG_MAPKW_NULL_SEGMENT), 2471 ld_map_tokenstr(tok, tkv, &inv_buf)); 2472 } 2473 2474 /* 2475 * Top Level Directive: 2476 * 2477 * NULL_SEGMENT segment_name { ... 2478 * ------------^ 2479 */ 2480 static Token 2481 dir_null_segment(Mapfile *mf) 2482 { 2483 return (dir_segment_inner(mf, MSG_ORIG(MSG_MAPKW_NULL_SEGMENT), 2484 PT_NULL, segment_core_attr_list, segment_core_attr_list_bufsize, 2485 gts_efunc_dir_null_segment)); 2486 2487 } 2488 2489 /* 2490 * Top Level Directive: 2491 * 2492 * SEGMENT_ORDER segment_name ... ; 2493 */ 2494 static Token 2495 dir_segment_order(Mapfile *mf) 2496 { 2497 Token tok; 2498 ld_map_tkval_t tkv; 2499 Conv_inv_buf_t inv_buf; 2500 Aliste idx; 2501 Sg_desc *sgp, *sgp2; 2502 int done; 2503 2504 /* Expect either a '=' or '+=' */ 2505 tok = gettoken_eq(mf, ATTR_FMT_EQ_PEQ, 2506 MSG_ORIG(MSG_MAPKW_SEGMENT_ORDER)); 2507 if (tok == TK_ERROR) 2508 return (TK_ERROR); 2509 2510 DBG_CALL(Dbg_map_seg_order(mf->mf_ofl, ELFOSABI_SOLARIS, 2511 ld_targ.t_m.m_mach, DBG_STATE_MOD_BEFORE, mf->mf_lineno)); 2512 2513 /* 2514 * The '=' form of assignment resets the list. The list contains 2515 * pointers to our mapfile text, so we do not have to free anything. 2516 */ 2517 if (tok == TK_EQUAL) 2518 aplist_reset(mf->mf_ofl->ofl_segs_order); 2519 2520 /* Read segment names, and add to list until terminator (';') is seen */ 2521 for (done = 0; done == 0; ) { 2522 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 2523 case TK_ERROR: 2524 return (TK_ERROR); 2525 2526 case TK_STRING: 2527 /* 2528 * The segment must have already been defined. 2529 */ 2530 sgp = ld_seg_lookup(mf->mf_ofl, tkv.tkv_str, NULL); 2531 if (sgp == NULL) { 2532 mf_fatal(mf, MSG_INTL(MSG_MAP_UNKSEG), 2533 tkv.tkv_str); 2534 return (TK_ERROR); 2535 } 2536 2537 /* 2538 * Make sure it's not already on the list 2539 */ 2540 for (APLIST_TRAVERSE(mf->mf_ofl->ofl_segs_order, 2541 idx, sgp2)) 2542 if (sgp == sgp2) { 2543 mf_fatal(mf, 2544 MSG_INTL(MSG_MAP_DUPORDSEG), 2545 MSG_ORIG(MSG_MAPKW_SEGMENT_ORDER), 2546 tkv.tkv_str); 2547 return (TK_ERROR); 2548 } 2549 2550 /* 2551 * It can't be ordered and also have an explicit 2552 * paddr or vaddr. 2553 */ 2554 if (sgp->sg_flags & (FLG_SG_P_PADDR | FLG_SG_P_VADDR)) { 2555 mf_fatal(mf, MSG_INTL(MSG_MAP_CNTADDRORDER), 2556 sgp->sg_name); 2557 return (TK_ERROR); 2558 } 2559 2560 2561 /* Put it at the end of the list */ 2562 if (aplist_append(&mf->mf_ofl->ofl_segs_order, sgp, 2563 AL_CNT_SG_IS_ORDER) == NULL) 2564 return (TK_ERROR); 2565 break; 2566 2567 case TK_SEMICOLON: 2568 done = 1; 2569 break; 2570 2571 default: 2572 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SEGNAM), 2573 MSG_ORIG(MSG_MAPKW_SEGMENT_ORDER), 2574 ld_map_tokenstr(tok, &tkv, &inv_buf)); 2575 return (TK_ERROR); 2576 } 2577 } 2578 2579 DBG_CALL(Dbg_map_seg_order(mf->mf_ofl, ELFOSABI_SOLARIS, 2580 ld_targ.t_m.m_mach, DBG_STATE_MOD_AFTER, mf->mf_lineno)); 2581 2582 return (tok); 2583 } 2584 2585 /* 2586 * Top Level Directive: 2587 * 2588 * STACK { ... 2589 * -----^ 2590 */ 2591 static Token 2592 dir_stack(Mapfile *mf) 2593 { 2594 /* STACK attributes */ 2595 static attr_t attr_list[] = { 2596 { MSG_ORIG(MSG_MAPKW_FLAGS), at_seg_flags, ATTR_FMT_EQ_ALL }, 2597 2598 /* List must be null terminated */ 2599 { 0 } 2600 }; 2601 2602 /* 2603 * Size of buffer needed to format the names in attr_list[]. Must 2604 * be kept in sync with attr_list. 2605 */ 2606 static size_t attr_list_bufsize = 2607 KW_NAME_SIZE(MSG_MAPKW_FLAGS); 2608 2609 Sg_desc *sgp; 2610 Token tok; 2611 2612 2613 /* Opening '{' token */ 2614 if (gettoken_leftbkt(mf, MSG_ORIG(MSG_MAPKW_STACK)) == TK_ERROR) 2615 return (TK_ERROR); 2616 2617 /* Fetch the PT_SUNWSTACK segment descriptor */ 2618 sgp = ld_map_seg_stack(mf); 2619 2620 /* Parse the attributes */ 2621 if (parse_attributes(mf, MSG_ORIG(MSG_MAPKW_STACK), 2622 attr_list, attr_list_bufsize, sgp) == TK_ERROR) 2623 return (TK_ERROR); 2624 2625 /* Terminating ';' */ 2626 tok = gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_STACK)); 2627 if (tok == TK_ERROR) 2628 return (TK_ERROR); 2629 2630 if (DBG_ENABLED) { 2631 Xword ndx = ld_map_seg_index(mf, sgp); 2632 2633 Dbg_map_seg(mf->mf_ofl, DBG_STATE_MOD_AFTER, ndx, sgp, 2634 mf->mf_lineno); 2635 } 2636 2637 return (tok); 2638 } 2639 2640 /* 2641 * at_sym_aux(): Value for AUXILIARY= is not an object name 2642 */ 2643 static void 2644 gts_efunc_at_sym_aux(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2645 { 2646 Conv_inv_buf_t inv_buf; 2647 2648 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_OBJNAM), 2649 MSG_ORIG(MSG_MAPKW_AUX), ld_map_tokenstr(tok, tkv, &inv_buf)); 2650 } 2651 2652 /* 2653 * SYMBOL [version_name] { symbol_name { AUXILIARY = soname 2654 * -------------------------------------------------^ 2655 */ 2656 /* ARGSUSED 1 */ 2657 static Token 2658 at_sym_aux(Mapfile *mf, Token eq_tok, void *uvalue) 2659 { 2660 symbol_state_t *ss = uvalue; 2661 ld_map_tkval_t tkv; 2662 2663 /* auxiliary filter soname */ 2664 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_sym_aux) == TK_ERROR) 2665 return (TK_ERROR); 2666 2667 ld_map_sym_filtee(mf, &ss->ss_mv, &ss->ss_ms, FLG_SY_AUXFLTR, 2668 tkv.tkv_str); 2669 2670 /* terminator */ 2671 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_AUX))); 2672 } 2673 2674 /* 2675 * at_sym_filter(): Value for FILTER= is not an object name 2676 */ 2677 static void 2678 gts_efunc_at_sym_filter(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2679 { 2680 Conv_inv_buf_t inv_buf; 2681 2682 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_OBJNAM), 2683 MSG_ORIG(MSG_MAPKW_FILTER), ld_map_tokenstr(tok, tkv, &inv_buf)); 2684 } 2685 2686 /* 2687 * SYMBOL [version_name] { symbol_name { FILTER = soname 2688 * ----------------------------------------------^ 2689 */ 2690 /* ARGSUSED 1 */ 2691 static Token 2692 at_sym_filter(Mapfile *mf, Token eq_tok, void *uvalue) 2693 { 2694 symbol_state_t *ss = uvalue; 2695 ld_map_tkval_t tkv; 2696 2697 /* filter soname */ 2698 if (gettoken_str(mf, 0, &tkv, gts_efunc_at_sym_filter) == TK_ERROR) 2699 return (TK_ERROR); 2700 2701 ld_map_sym_filtee(mf, &ss->ss_mv, &ss->ss_ms, FLG_SY_STDFLTR, 2702 tkv.tkv_str); 2703 2704 /* terminator */ 2705 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_FILTER))); 2706 } 2707 2708 /* 2709 * SYMBOL [version_name] { symbol_name { FLAGS = ... 2710 * ---------------------------------------------^ 2711 */ 2712 /* ARGSUSED 1 */ 2713 static Token 2714 at_sym_flags(Mapfile *mf, Token eq_tok, void *uvalue) 2715 { 2716 typedef struct { 2717 const char *name; 2718 sd_flag_t value; 2719 } symflag_t; 2720 2721 static symflag_t symflag_list[] = { 2722 { MSG_ORIG(MSG_MAPKW_DIRECT), FLG_SY_DIR }, 2723 { MSG_ORIG(MSG_MAPKW_DYNSORT), FLG_SY_DYNSORT }, 2724 { MSG_ORIG(MSG_MAPKW_EXTERN), FLG_SY_EXTERN }, 2725 { MSG_ORIG(MSG_MAPKW_INTERPOSE), FLG_SY_INTPOSE }, 2726 { MSG_ORIG(MSG_MAPKW_NODIRECT), FLG_SY_NDIR }, 2727 { MSG_ORIG(MSG_MAPKW_NODYNSORT), FLG_SY_NODYNSORT }, 2728 { MSG_ORIG(MSG_MAPKW_PARENT), FLG_SY_PARENT }, 2729 2730 /* List must be null terminated */ 2731 { 0 } 2732 }; 2733 2734 /* 2735 * Size of buffer needed to format the names in flag_list[]. Must 2736 * be kept in sync with flag_list. 2737 */ 2738 static size_t symflag_list_bufsize = 2739 KW_NAME_SIZE(MSG_MAPKW_DIRECT) + 2740 KW_NAME_SIZE(MSG_MAPKW_DYNSORT) + 2741 KW_NAME_SIZE(MSG_MAPKW_EXTERN) + 2742 KW_NAME_SIZE(MSG_MAPKW_INTERPOSE) + 2743 KW_NAME_SIZE(MSG_MAPKW_NODIRECT) + 2744 KW_NAME_SIZE(MSG_MAPKW_NODYNSORT) + 2745 KW_NAME_SIZE(MSG_MAPKW_PARENT); 2746 2747 symbol_state_t *ss = uvalue; 2748 int done; 2749 symflag_t *symflag; 2750 int cnt = 0; 2751 Token tok; 2752 ld_map_tkval_t tkv; 2753 Conv_inv_buf_t inv_buf; 2754 Ofl_desc *ofl = mf->mf_ofl; 2755 2756 for (done = 0; done == 0; ) { 2757 switch (tok = ld_map_gettoken(mf, TK_F_KEYWORD, &tkv)) { 2758 case TK_ERROR: 2759 return (TK_ERROR); 2760 2761 case TK_STRING: 2762 symflag = ld_map_kwfind(tkv.tkv_str, symflag_list, 2763 SGSOFFSETOF(symflag_t, name), sizeof (symflag[0])); 2764 if (symflag == NULL) 2765 goto bad_flag; 2766 cnt++; 2767 /* 2768 * Apply the flag: 2769 * 2770 * Although tempting to make all of this table-driven 2771 * via added fields in symflag_t, there's enough 2772 * variation in what each flag does to make that 2773 * not quite worthwhile. 2774 * 2775 * Similarly, it is tempting to use common code to 2776 * to do this work from map_support.c. However, the 2777 * v1 code mixes unrelated things (flags, symbol types, 2778 * value, size, etc) in single cascading series of 2779 * strcmps, whereas our parsing separates those things 2780 * from each other. Merging the code would require doing 2781 * two strcmps for each item, or other complexity, 2782 * which I judge not to be worthwhile. 2783 */ 2784 switch (symflag->value) { 2785 case FLG_SY_DIR: 2786 ss->ss_ms.ms_sdflags |= FLG_SY_DIR; 2787 ofl->ofl_flags |= FLG_OF_SYMINFO; 2788 break; 2789 case FLG_SY_DYNSORT: 2790 ss->ss_ms.ms_sdflags |= FLG_SY_DYNSORT; 2791 ss->ss_ms.ms_sdflags &= ~FLG_SY_NODYNSORT; 2792 break; 2793 case FLG_SY_EXTERN: 2794 ss->ss_ms.ms_sdflags |= FLG_SY_EXTERN; 2795 ofl->ofl_flags |= FLG_OF_SYMINFO; 2796 break; 2797 case FLG_SY_INTPOSE: 2798 if (!(ofl->ofl_flags & FLG_OF_EXEC)) { 2799 mf_fatal0(mf, 2800 MSG_INTL(MSG_MAP_NOINTPOSE)); 2801 ss->ss_mv.mv_errcnt++; 2802 break; 2803 } 2804 ss->ss_ms.ms_sdflags |= FLG_SY_INTPOSE; 2805 ofl->ofl_flags |= FLG_OF_SYMINFO; 2806 ofl->ofl_dtflags_1 |= DF_1_SYMINTPOSE; 2807 break; 2808 case FLG_SY_NDIR: 2809 ss->ss_ms.ms_sdflags |= FLG_SY_NDIR; 2810 ofl->ofl_flags |= FLG_OF_SYMINFO; 2811 ofl->ofl_flags1 |= 2812 (FLG_OF1_NDIRECT | FLG_OF1_NGLBDIR); 2813 break; 2814 case FLG_SY_NODYNSORT: 2815 ss->ss_ms.ms_sdflags &= ~FLG_SY_DYNSORT; 2816 ss->ss_ms.ms_sdflags |= FLG_SY_NODYNSORT; 2817 break; 2818 case FLG_SY_PARENT: 2819 ss->ss_ms.ms_sdflags |= FLG_SY_PARENT; 2820 ofl->ofl_flags |= FLG_OF_SYMINFO; 2821 break; 2822 } 2823 break; 2824 case TK_RIGHTBKT: 2825 case TK_SEMICOLON: 2826 done = 1; 2827 break; 2828 2829 default: 2830 bad_flag: 2831 { 2832 char buf[VLA_SIZE(symflag_list_bufsize)]; 2833 2834 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SYMFLAG), 2835 ld_map_kwnames(symflag_list, 2836 SGSOFFSETOF(symflag_t, name), 2837 sizeof (symflag[0]), buf, 2838 symflag_list_bufsize), 2839 ld_map_tokenstr(tok, &tkv, &inv_buf)); 2840 } 2841 return (TK_ERROR); 2842 } 2843 } 2844 2845 /* Make sure there was at least one flag specified */ 2846 if (cnt == 0) { 2847 mf_fatal(mf, MSG_INTL(MSG_MAP_NOVALUES), 2848 MSG_ORIG(MSG_MAPKW_FLAGS)); 2849 return (TK_ERROR); 2850 } 2851 2852 return (tok); /* Either TK_SEMICOLON or TK_RIGHTBKT */ 2853 } 2854 2855 /* 2856 * SYMBOL [version_name] { symbol_name { SIZE = value 2857 * --------------------------------------------^ 2858 */ 2859 /* ARGSUSED 1 */ 2860 static Token 2861 at_sym_size(Mapfile *mf, Token eq_tok, void *uvalue) 2862 { 2863 symbol_state_t *ss = uvalue; 2864 ld_map_tkval_t tkv; 2865 2866 /* value */ 2867 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_SIZE), &tkv) == TK_ERROR) 2868 return (TK_ERROR); 2869 2870 ss->ss_ms.ms_size = tkv.tkv_int.tkvi_value; 2871 2872 /* terminator */ 2873 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_SIZE))); 2874 } 2875 2876 typedef struct { 2877 const char *name; /* type name */ 2878 Word ms_shndx; /* symbol section index */ 2879 uchar_t ms_type; /* STT_ symbol type */ 2880 } at_sym_type_t; 2881 2882 static at_sym_type_t at_sym_type_list[] = { 2883 { MSG_ORIG(MSG_MAPKW_COMMON), SHN_COMMON, STT_OBJECT }, 2884 { MSG_ORIG(MSG_MAPKW_DATA), SHN_ABS, STT_OBJECT }, 2885 { MSG_ORIG(MSG_MAPKW_FUNCTION), SHN_ABS, STT_FUNC }, 2886 2887 /* List must be null terminated */ 2888 { 0 } 2889 }; 2890 2891 /* 2892 * Size of buffer needed to format the names in at_sym_type_list[]. Must 2893 * be kept in sync with at_sym_type_list. 2894 */ 2895 static size_t at_sym_type_list_bufsize = 2896 KW_NAME_SIZE(MSG_MAPKW_COMMON) + 2897 KW_NAME_SIZE(MSG_MAPKW_DATA) + 2898 KW_NAME_SIZE(MSG_MAPKW_FUNCTION); 2899 2900 /* 2901 * at_sym_type(): Value for TYPE= is not a symbol type 2902 */ 2903 static void 2904 gts_efunc_at_sym_type(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 2905 { 2906 Conv_inv_buf_t inv_buf; 2907 char buf[VLA_SIZE(at_sym_type_list_bufsize)]; 2908 2909 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SYMTYPE), 2910 ld_map_kwnames(at_sym_type_list, SGSOFFSETOF(at_sym_type_t, name), 2911 sizeof (at_sym_type_list[0]), buf, at_sym_type_list_bufsize), 2912 ld_map_tokenstr(tok, tkv, &inv_buf)); 2913 } 2914 2915 /* 2916 * SYMBOL [version_name] { symbol_name { TYPE = symbol_type 2917 * --------------------------------------------^ 2918 */ 2919 /* ARGSUSED 1 */ 2920 static Token 2921 at_sym_type(Mapfile *mf, Token eq_tok, void *uvalue) 2922 { 2923 symbol_state_t *ss = uvalue; 2924 at_sym_type_t *type; 2925 ld_map_tkval_t tkv; 2926 2927 /* type keyword */ 2928 if (gettoken_str(mf, TK_F_KEYWORD, &tkv, gts_efunc_at_sym_type) == 2929 TK_ERROR) 2930 return (TK_ERROR); 2931 2932 type = ld_map_kwfind(tkv.tkv_str, at_sym_type_list, 2933 SGSOFFSETOF(at_sym_type_t, name), sizeof (type[0])); 2934 if (type == NULL) { 2935 gts_efunc_at_sym_type(mf, TK_STRING, &tkv); 2936 return (TK_ERROR); 2937 } 2938 2939 ss->ss_ms.ms_shndx = type->ms_shndx; 2940 ss->ss_ms.ms_sdflags |= FLG_SY_SPECSEC; 2941 ss->ss_ms.ms_type = type->ms_type; 2942 2943 /* terminator */ 2944 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_TYPE))); 2945 } 2946 2947 /* 2948 * SYMBOL [version_name] { symbol_name { VALUE = value 2949 * ---------------------------------------------^ 2950 */ 2951 /* ARGSUSED 1 */ 2952 static Token 2953 at_sym_value(Mapfile *mf, Token eq_tok, void *uvalue) 2954 { 2955 symbol_state_t *ss = uvalue; 2956 ld_map_tkval_t tkv; 2957 2958 /* value */ 2959 if (gettoken_int(mf, MSG_ORIG(MSG_MAPKW_VALUE), &tkv) == TK_ERROR) 2960 return (TK_ERROR); 2961 2962 ss->ss_ms.ms_value = tkv.tkv_int.tkvi_value; 2963 ss->ss_ms.ms_value_set = TRUE; 2964 2965 2966 /* terminator */ 2967 return (gettoken_term(mf, MSG_ORIG(MSG_MAPKW_VALUE))); 2968 } 2969 2970 /* 2971 * Parse the attributes for a SCOPE or VERSION symbol directive. 2972 * 2973 * entry: 2974 * mf - Mapfile descriptor 2975 * dir_name - Name of directive. 2976 * ss - Pointer to symbol state block that has had its ss_nv 2977 * member initialzed via a call to ld_map_sym_ver_init(). 2978 * 2979 * exit: 2980 * parse_symbol_attributes() returns TK_RIGHTBKT on success, and TK_ERROR 2981 * on failure. 2982 */ 2983 static Token 2984 parse_symbol_attributes(Mapfile *mf, const char *dir_name, symbol_state_t *ss) 2985 { 2986 /* Symbol attributes */ 2987 static attr_t attr_list[] = { 2988 { MSG_ORIG(MSG_MAPKW_AUX), at_sym_aux, ATTR_FMT_EQ }, 2989 { MSG_ORIG(MSG_MAPKW_FILTER), at_sym_filter, ATTR_FMT_EQ }, 2990 { MSG_ORIG(MSG_MAPKW_FLAGS), at_sym_flags, ATTR_FMT_EQ }, 2991 { MSG_ORIG(MSG_MAPKW_SIZE), at_sym_size, ATTR_FMT_EQ }, 2992 { MSG_ORIG(MSG_MAPKW_TYPE), at_sym_type, ATTR_FMT_EQ }, 2993 { MSG_ORIG(MSG_MAPKW_VALUE), at_sym_value, ATTR_FMT_EQ }, 2994 2995 /* List must be null terminated */ 2996 { 0 } 2997 }; 2998 2999 /* 3000 * Size of buffer needed to format the names in attr_list[]. Must 3001 * be kept in sync with attr_list. 3002 */ 3003 static size_t attr_list_bufsize = 3004 KW_NAME_SIZE(MSG_MAPKW_AUX) + 3005 KW_NAME_SIZE(MSG_MAPKW_FILTER) + 3006 KW_NAME_SIZE(MSG_MAPKW_FLAGS) + 3007 KW_NAME_SIZE(MSG_MAPKW_SIZE) + 3008 KW_NAME_SIZE(MSG_MAPKW_TYPE) + 3009 KW_NAME_SIZE(MSG_MAPKW_VALUE); 3010 3011 Token tok; 3012 ld_map_tkval_t tkv, tkv_sym; 3013 int done; 3014 Conv_inv_buf_t inv_buf; 3015 3016 /* Read attributes until the closing '}' is seen */ 3017 for (done = 0; done == 0; ) { 3018 /* 3019 * We have to allow quotes around symbol names, but the 3020 * name we read may also be a symbol scope keyword. We won't 3021 * know which until we read the following token, and so have 3022 * to allow quotes for both. Hence, symbol scope names can 3023 * be quoted --- an unlikely occurrence and not worth 3024 * complicating the code. 3025 */ 3026 switch (tok = ld_map_gettoken(mf, 0, &tkv_sym)) { 3027 case TK_ERROR: 3028 return (TK_ERROR); 3029 3030 case TK_STRING: 3031 /* Default value for all symbol attributes is 0 */ 3032 (void) memset(&ss->ss_ms, 0, sizeof (ss->ss_ms)); 3033 ss->ss_ms.ms_name = tkv_sym.tkv_str; 3034 3035 /* 3036 * Turn off the WEAK flag to indicate that definitions 3037 * are associated with this version. It would probably 3038 * be more accurate to only remove this flag with the 3039 * specification of global symbols, however setting it 3040 * here allows enough slop to compensate for the 3041 * various user inputs we've seen so far. Only if a 3042 * closed version is specified (i.e., "SUNW_1.x {};") 3043 * will a user get a weak version (which is how we 3044 * document the creation of weak versions). 3045 */ 3046 ss->ss_mv.mv_vdp->vd_flags &= ~VER_FLG_WEAK; 3047 3048 /* 3049 * The meaning of this name depends on the following 3050 * character: 3051 * 3052 * : Scope 3053 * ; Symbol without attributes 3054 * { Symbol with attributes 3055 */ 3056 switch (tok = ld_map_gettoken(mf, 0, &tkv)) { 3057 case TK_ERROR: 3058 return (TK_ERROR); 3059 3060 case TK_COLON: 3061 ld_map_sym_scope(mf, tkv_sym.tkv_str, 3062 &ss->ss_mv); 3063 break; 3064 case TK_LEFTBKT: 3065 /* name is a symbol with attributes */ 3066 if (parse_attributes(mf, tkv_sym.tkv_str, 3067 attr_list, attr_list_bufsize, ss) == 3068 TK_ERROR) 3069 return (TK_ERROR); 3070 /* Terminating ';', or '}' */ 3071 tok = gettoken_term(mf, 3072 MSG_INTL(MSG_MAP_SYMATTR)); 3073 if (tok == TK_ERROR) 3074 return (TK_ERROR); 3075 if (tok == TK_RIGHTBKT) 3076 done = 1; 3077 3078 /* FALLTHROUGH */ 3079 case TK_SEMICOLON: 3080 /* 3081 * Add the new symbol. It should be noted that 3082 * all symbols added by the mapfile start out 3083 * with global scope, thus they will fall 3084 * through the normal symbol resolution 3085 * process. Symbols defined as locals will 3086 * be reduced in scope after all input file 3087 * processing. 3088 */ 3089 if (!ld_map_sym_enter(mf, &ss->ss_mv, 3090 &ss->ss_ms)) 3091 return (TK_ERROR); 3092 break; 3093 default: 3094 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SYMDELIM), 3095 ld_map_tokenstr(tok, &tkv, &inv_buf)); 3096 return (TK_ERROR); 3097 } 3098 break; 3099 3100 case TK_RIGHTBKT: 3101 done = 1; 3102 break; 3103 3104 case TK_SEMICOLON: 3105 break; /* Ignore empty statement */ 3106 3107 case TK_STAR: 3108 /* 3109 * Turn off the WEAK flag, as explained above for 3110 * TK_STRING. 3111 */ 3112 ss->ss_mv.mv_vdp->vd_flags &= ~VER_FLG_WEAK; 3113 3114 ld_map_sym_autoreduce(mf, &ss->ss_mv); 3115 3116 /* 3117 * Following token must be ';' to terminate the stmt, 3118 * or '}' to terminate the whole directive. 3119 */ 3120 switch (tok = gettoken_term(mf, dir_name)) { 3121 case TK_ERROR: 3122 return (TK_ERROR); 3123 case TK_RIGHTBKT: 3124 done = 1; 3125 break; 3126 } 3127 break; 3128 3129 default: 3130 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_SYM), 3131 ld_map_tokenstr(tok, &tkv_sym, &inv_buf)); 3132 return (TK_ERROR); 3133 } 3134 } 3135 3136 /* 3137 * In the SYMBOL directive, we keep parsing in the face of 3138 * errors that don't involve resources, to maximize what we 3139 * can report in a single invocation. If we encountered such 3140 * an error, act on the error(s) now. 3141 */ 3142 if (ss->ss_mv.mv_errcnt) 3143 return (TK_ERROR); 3144 3145 return (tok); 3146 } 3147 3148 3149 /* 3150 * Top Level Directive: 3151 * 3152 * SYMBOL_SCOPE { ... 3153 * ------------^ 3154 */ 3155 static Token 3156 dir_symbol_scope(Mapfile *mf) 3157 { 3158 symbol_state_t ss; 3159 3160 /* The first token must be a '{' */ 3161 if (gettoken_leftbkt(mf, MSG_ORIG(MSG_MAPKW_SYMBOL_SCOPE)) == TK_ERROR) 3162 return (TK_ERROR); 3163 3164 /* Establish the version descriptor and related data */ 3165 if (!ld_map_sym_ver_init(mf, NULL, &ss.ss_mv)) 3166 return (TK_ERROR); 3167 3168 /* Read attributes until the closing '}' is seen */ 3169 if (parse_symbol_attributes(mf, MSG_ORIG(MSG_MAPKW_SYMBOL_SCOPE), 3170 &ss) == TK_ERROR) 3171 return (TK_ERROR); 3172 3173 /* Terminating ';' */ 3174 return (gettoken_semicolon(mf, MSG_ORIG(MSG_MAPKW_SYMBOL_SCOPE))); 3175 } 3176 3177 3178 /* 3179 * at_dv_allow(): Value for ALLOW= is not a version string 3180 */ 3181 static void 3182 gts_efunc_dir_symbol_version(Mapfile *mf, Token tok, ld_map_tkval_t *tkv) 3183 { 3184 Conv_inv_buf_t inv_buf; 3185 3186 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_VERSION), 3187 MSG_ORIG(MSG_MAPKW_SYMBOL_VERSION), 3188 ld_map_tokenstr(tok, tkv, &inv_buf)); 3189 } 3190 3191 /* 3192 * Top Level Directive: 3193 * 3194 * SYMBOL_VERSION version_name { ... 3195 * --------------^ 3196 */ 3197 static Token 3198 dir_symbol_version(Mapfile *mf) 3199 { 3200 3201 ld_map_tkval_t tkv; 3202 symbol_state_t ss; 3203 3204 /* The first token must be a version name */ 3205 if (gettoken_str(mf, 0, &tkv, gts_efunc_dir_symbol_version) == TK_ERROR) 3206 return (TK_ERROR); 3207 3208 /* The next token is expected to be '{' */ 3209 if (gettoken_leftbkt(mf, MSG_ORIG(MSG_MAPKW_SYMBOL_VERSION)) == 3210 TK_ERROR) 3211 return (TK_ERROR); 3212 3213 /* Establish the version descriptor and related data */ 3214 if (!ld_map_sym_ver_init(mf, tkv.tkv_str, &ss.ss_mv)) 3215 return (TK_ERROR); 3216 3217 /* Read attributes until the closing '}' is seen */ 3218 if (parse_symbol_attributes(mf, MSG_ORIG(MSG_MAPKW_SYMBOL_VERSION), 3219 &ss) == TK_ERROR) 3220 return (TK_ERROR); 3221 3222 /* 3223 * Determine if any version references are provided after the close 3224 * bracket, parsing up to the terminating ';'. 3225 */ 3226 if (!ld_map_sym_ver_fini(mf, &ss.ss_mv)) 3227 return (TK_ERROR); 3228 3229 return (TK_SEMICOLON); 3230 } 3231 3232 3233 /* 3234 * Parse the mapfile --- Solaris syntax 3235 */ 3236 Boolean 3237 ld_map_parse_v2(Mapfile *mf) 3238 { 3239 /* Valid top level mapfile directives */ 3240 typedef struct { 3241 const char *name; /* Directive */ 3242 dir_func_t func; /* Function to parse directive */ 3243 } tldir_t; 3244 3245 3246 tldir_t dirlist[] = { 3247 { MSG_ORIG(MSG_MAPKW_CAPABILITY), dir_capability }, 3248 { MSG_ORIG(MSG_MAPKW_DEPEND_VERSIONS), dir_depend_versions }, 3249 { MSG_ORIG(MSG_MAPKW_HDR_NOALLOC), dir_hdr_noalloc }, 3250 { MSG_ORIG(MSG_MAPKW_LOAD_SEGMENT), dir_load_segment }, 3251 { MSG_ORIG(MSG_MAPKW_NOTE_SEGMENT), dir_note_segment }, 3252 { MSG_ORIG(MSG_MAPKW_NULL_SEGMENT), dir_null_segment }, 3253 { MSG_ORIG(MSG_MAPKW_PHDR_ADD_NULL), dir_phdr_add_null }, 3254 { MSG_ORIG(MSG_MAPKW_SEGMENT_ORDER), dir_segment_order }, 3255 { MSG_ORIG(MSG_MAPKW_STACK), dir_stack }, 3256 { MSG_ORIG(MSG_MAPKW_SYMBOL_SCOPE), dir_symbol_scope }, 3257 { MSG_ORIG(MSG_MAPKW_SYMBOL_VERSION), dir_symbol_version }, 3258 3259 /* List must be null terminated */ 3260 { 0 } 3261 }; 3262 3263 /* 3264 * Size of buffer needed to format the names in dirlist[]. Must 3265 * be kept in sync with dirlist. 3266 */ 3267 static size_t dirlist_bufsize = 3268 KW_NAME_SIZE(MSG_MAPKW_CAPABILITY) + 3269 KW_NAME_SIZE(MSG_MAPKW_DEPEND_VERSIONS) + 3270 KW_NAME_SIZE(MSG_MAPKW_HDR_NOALLOC) + 3271 KW_NAME_SIZE(MSG_MAPKW_LOAD_SEGMENT) + 3272 KW_NAME_SIZE(MSG_MAPKW_NOTE_SEGMENT) + 3273 KW_NAME_SIZE(MSG_MAPKW_NULL_SEGMENT) + 3274 KW_NAME_SIZE(MSG_MAPKW_PHDR_ADD_NULL) + 3275 KW_NAME_SIZE(MSG_MAPKW_SEGMENT_ORDER) + 3276 KW_NAME_SIZE(MSG_MAPKW_STACK) + 3277 KW_NAME_SIZE(MSG_MAPKW_SYMBOL_SCOPE) + 3278 KW_NAME_SIZE(MSG_MAPKW_SYMBOL_VERSION); 3279 3280 Token tok; /* current token. */ 3281 ld_map_tkval_t tkv; /* Value of token */ 3282 tldir_t *tldir; 3283 Conv_inv_buf_t inv_buf; 3284 3285 for (;;) { 3286 tok = ld_map_gettoken(mf, TK_F_EOFOK | TK_F_KEYWORD, &tkv); 3287 switch (tok) { 3288 case TK_ERROR: 3289 return (FALSE); 3290 case TK_EOF: 3291 return (TRUE); 3292 case TK_SEMICOLON: /* Terminator, or empty directive: Ignore */ 3293 break; 3294 case TK_STRING: 3295 /* Map name to entry in dirlist[] */ 3296 tldir = ld_map_kwfind(tkv.tkv_str, dirlist, 3297 SGSOFFSETOF(tldir_t, name), sizeof (dirlist[0])); 3298 3299 /* Not a directive we know? */ 3300 if (tldir == NULL) 3301 goto bad_dirtok; 3302 3303 /* Call the function associated with this directive */ 3304 if (tldir->func(mf) == TK_ERROR) 3305 return (FALSE); 3306 break; 3307 default: 3308 bad_dirtok: 3309 { 3310 char buf[VLA_SIZE(dirlist_bufsize)]; 3311 3312 mf_fatal(mf, MSG_INTL(MSG_MAP_EXP_DIR), 3313 ld_map_kwnames(dirlist, 3314 SGSOFFSETOF(tldir_t, name), 3315 sizeof (dirlist[0]), buf, dirlist_bufsize), 3316 ld_map_tokenstr(tok, &tkv, &inv_buf)); 3317 } 3318 return (FALSE); 3319 } 3320 } 3321 3322 /*NOTREACHED*/ 3323 assert(0); 3324 return (FALSE); 3325 }