|  | /* expr.h -> header file for expr.c | 
|  | Copyright (C) 1987-2021 Free Software Foundation, Inc. | 
|  |  | 
|  | This file is part of GAS, the GNU Assembler. | 
|  |  | 
|  | GAS is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3, or (at your option) | 
|  | any later version. | 
|  |  | 
|  | GAS is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with GAS; see the file COPYING.  If not, write to the Free | 
|  | Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA | 
|  | 02110-1301, USA.  */ | 
|  |  | 
|  | /* | 
|  | * By popular demand, we define a struct to represent an expression. | 
|  | * This will no doubt mutate as expressions become baroque. | 
|  | * | 
|  | * Currently, we support expressions like "foo OP bar + 42".  In other | 
|  | * words we permit a (possibly undefined) symbol, a (possibly | 
|  | * undefined) symbol and the operation used to combine the symbols, | 
|  | * and an (absolute) augend.  RMS says this is so we can have 1-pass | 
|  | * assembly for any compiler emissions, and a 'case' statement might | 
|  | * emit 'undefined1 - undefined2'. | 
|  | * | 
|  | * The type of an expression used to be stored as a segment.  That got | 
|  | * confusing because it overloaded the concept of a segment.  I added | 
|  | * an operator field, instead. | 
|  | */ | 
|  |  | 
|  | /* This is the type of an expression.  The operator types are also | 
|  | used while parsing an expression. | 
|  |  | 
|  | NOTE: This enumeration must match the op_rank array in expr.c.  */ | 
|  |  | 
|  | typedef enum { | 
|  | /* An illegal expression.  */ | 
|  | O_illegal, | 
|  | /* A nonexistent expression.  */ | 
|  | O_absent, | 
|  | /* X_add_number (a constant expression).  */ | 
|  | O_constant, | 
|  | /* X_add_symbol + X_add_number.  */ | 
|  | O_symbol, | 
|  | /* X_add_symbol + X_add_number - the base address of the image.  */ | 
|  | O_symbol_rva, | 
|  | /* A register (X_add_number is register number).  */ | 
|  | O_register, | 
|  | /* A big value.  If X_add_number is negative or 0, the value is in | 
|  | generic_floating_point_number.  Otherwise the value is in | 
|  | generic_bignum, and X_add_number is the number of LITTLENUMs in | 
|  | the value.  */ | 
|  | O_big, | 
|  | /* (- X_add_symbol) + X_add_number.  */ | 
|  | O_uminus, | 
|  | /* (~ X_add_symbol) + X_add_number.  */ | 
|  | O_bit_not, | 
|  | /* (! X_add_symbol) + X_add_number.  */ | 
|  | O_logical_not, | 
|  | /* (X_add_symbol * X_op_symbol) + X_add_number.  */ | 
|  | O_multiply, | 
|  | /* (X_add_symbol / X_op_symbol) + X_add_number.  */ | 
|  | O_divide, | 
|  | /* (X_add_symbol % X_op_symbol) + X_add_number.  */ | 
|  | O_modulus, | 
|  | /* (X_add_symbol << X_op_symbol) + X_add_number.  */ | 
|  | O_left_shift, | 
|  | /* (X_add_symbol >> X_op_symbol) + X_add_number.  */ | 
|  | O_right_shift, | 
|  | /* (X_add_symbol | X_op_symbol) + X_add_number.  */ | 
|  | O_bit_inclusive_or, | 
|  | /* (X_add_symbol |~ X_op_symbol) + X_add_number.  */ | 
|  | O_bit_or_not, | 
|  | /* (X_add_symbol ^ X_op_symbol) + X_add_number.  */ | 
|  | O_bit_exclusive_or, | 
|  | /* (X_add_symbol & X_op_symbol) + X_add_number.  */ | 
|  | O_bit_and, | 
|  | /* (X_add_symbol + X_op_symbol) + X_add_number.  */ | 
|  | O_add, | 
|  | /* (X_add_symbol - X_op_symbol) + X_add_number.  */ | 
|  | O_subtract, | 
|  | /* (X_add_symbol == X_op_symbol) + X_add_number.  */ | 
|  | O_eq, | 
|  | /* (X_add_symbol != X_op_symbol) + X_add_number.  */ | 
|  | O_ne, | 
|  | /* (X_add_symbol < X_op_symbol) + X_add_number.  */ | 
|  | O_lt, | 
|  | /* (X_add_symbol <= X_op_symbol) + X_add_number.  */ | 
|  | O_le, | 
|  | /* (X_add_symbol >= X_op_symbol) + X_add_number.  */ | 
|  | O_ge, | 
|  | /* (X_add_symbol > X_op_symbol) + X_add_number.  */ | 
|  | O_gt, | 
|  | /* (X_add_symbol && X_op_symbol) + X_add_number.  */ | 
|  | O_logical_and, | 
|  | /* (X_add_symbol || X_op_symbol) + X_add_number.  */ | 
|  | O_logical_or, | 
|  | /* X_op_symbol [ X_add_symbol ] */ | 
|  | O_index, | 
|  | /* machine dependent operators */ | 
|  | O_md1,  O_md2,  O_md3,  O_md4,  O_md5,  O_md6,  O_md7,  O_md8, | 
|  | O_md9,  O_md10, O_md11, O_md12, O_md13, O_md14, O_md15, O_md16, | 
|  | O_md17, O_md18, O_md19, O_md20, O_md21, O_md22, O_md23, O_md24, | 
|  | O_md25, O_md26, O_md27, O_md28, O_md29, O_md30, O_md31, O_md32, | 
|  | /* this must be the largest value */ | 
|  | O_max | 
|  | } operatorT; | 
|  |  | 
|  | typedef struct expressionS { | 
|  | /* The main symbol.  */ | 
|  | symbolS *X_add_symbol; | 
|  | /* The second symbol, if needed.  */ | 
|  | symbolS *X_op_symbol; | 
|  | /* A number to add.  */ | 
|  | offsetT X_add_number; | 
|  |  | 
|  | /* The type of the expression.  We can't assume that an arbitrary | 
|  | compiler can handle a bitfield of enum type.  FIXME: We could | 
|  | check this using autoconf.  */ | 
|  | #ifdef __GNUC__ | 
|  | operatorT X_op : 8; | 
|  | #else | 
|  | unsigned char X_op; | 
|  | #endif | 
|  |  | 
|  | /* Non-zero if X_add_number should be regarded as unsigned.  This is | 
|  | only valid for O_constant expressions.  It is only used when an | 
|  | O_constant must be extended into a bignum (i.e., it is not used | 
|  | when performing arithmetic on these values). | 
|  | FIXME: This field is not set very reliably.  */ | 
|  | unsigned int X_unsigned : 1; | 
|  | /* This is used to implement "word size + 1 bit" arithmetic, so that e.g. | 
|  | expressions used with .sleb128 directives can use the full range available | 
|  | for an unsigned word, but can also properly represent all values of a | 
|  | signed word.  */ | 
|  | unsigned int X_extrabit : 1; | 
|  |  | 
|  | /* 7 additional bits can be defined if needed.  */ | 
|  |  | 
|  | /* Machine dependent field */ | 
|  | unsigned short X_md; | 
|  | } expressionS; | 
|  |  | 
|  | enum expr_mode | 
|  | { | 
|  | expr_evaluate, | 
|  | expr_normal, | 
|  | expr_defer | 
|  | }; | 
|  |  | 
|  | /* "result" should be type (expressionS *).  */ | 
|  | #define expression(result) expr (0, result, expr_normal) | 
|  | #define expression_and_evaluate(result) expr (0, result, expr_evaluate) | 
|  | #define deferred_expression(result) expr (0, result, expr_defer) | 
|  |  | 
|  | /* If an expression is O_big, look here for its value. These common | 
|  | data may be clobbered whenever expr() is called.  */ | 
|  | /* Flonums returned here.  Big enough to hold most precise flonum.  */ | 
|  | extern FLONUM_TYPE generic_floating_point_number; | 
|  | /* Bignums returned here.  */ | 
|  | extern LITTLENUM_TYPE generic_bignum[]; | 
|  | /* Number of littlenums in above.  */ | 
|  | #define SIZE_OF_LARGE_NUMBER (20) | 
|  |  | 
|  | typedef char operator_rankT; | 
|  |  | 
|  | extern char get_symbol_name (char **); | 
|  | extern char restore_line_pointer (char); | 
|  | extern void expr_begin (void); | 
|  | extern void expr_set_precedence (void); | 
|  | extern void expr_set_rank (operatorT, operator_rankT); | 
|  | extern void add_to_result (expressionS *, offsetT, int); | 
|  | extern void subtract_from_result (expressionS *, offsetT, int); | 
|  | extern segT expr (int, expressionS *, enum expr_mode); | 
|  | extern unsigned int get_single_number (void); | 
|  | extern symbolS *make_expr_symbol (expressionS * expressionP); | 
|  | extern int expr_symbol_where (symbolS *, const char **, unsigned int *); | 
|  | extern void current_location (expressionS *); | 
|  |  | 
|  | extern symbolS *expr_build_uconstant (offsetT); | 
|  | extern symbolS *expr_build_dot (void); | 
|  | extern uint32_t generic_bignum_to_int32 (void); | 
|  | extern uint64_t generic_bignum_to_int64 (void); | 
|  |  | 
|  | int resolve_expression (expressionS *); | 
|  |  | 
|  | extern bool literal_prefix_dollar_hex; |