Home | History | Annotate | Line # | Download | only in gcc
      1 /* Declarations and definitions dealing with attribute handling.
      2    Copyright (C) 2013-2022 Free Software Foundation, Inc.
      3 
      4 This file is part of GCC.
      5 
      6 GCC is free software; you can redistribute it and/or modify it under
      7 the terms of the GNU General Public License as published by the Free
      8 Software Foundation; either version 3, or (at your option) any later
      9 version.
     10 
     11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
     12 WARRANTY; without even the implied warranty of MERCHANTABILITY or
     13 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
     14 for more details.
     15 
     16 You should have received a copy of the GNU General Public License
     17 along with GCC; see the file COPYING3.  If not see
     18 <http://www.gnu.org/licenses/>.  */
     19 
     20 #ifndef GCC_ATTRIBS_H
     21 #define GCC_ATTRIBS_H
     22 
     23 extern const struct attribute_spec *lookup_attribute_spec (const_tree);
     24 extern void free_attr_data ();
     25 extern void init_attributes (void);
     26 
     27 /* Process the attributes listed in ATTRIBUTES and install them in *NODE,
     28    which is either a DECL (including a TYPE_DECL) or a TYPE.  If a DECL,
     29    it should be modified in place; if a TYPE, a copy should be created
     30    unless ATTR_FLAG_TYPE_IN_PLACE is set in FLAGS.  FLAGS gives further
     31    information, in the form of a bitwise OR of flags in enum attribute_flags
     32    from tree.h.  Depending on these flags, some attributes may be
     33    returned to be applied at a later stage (for example, to apply
     34    a decl attribute to the declaration rather than to its type).  */
     35 extern tree decl_attributes (tree *, tree, int, tree = NULL_TREE);
     36 
     37 extern bool cxx11_attribute_p (const_tree);
     38 extern tree get_attribute_name (const_tree);
     39 extern tree get_attribute_namespace (const_tree);
     40 extern void apply_tm_attr (tree, tree);
     41 extern tree make_attribute (const char *, const char *, tree);
     42 extern bool attribute_ignored_p (tree);
     43 extern bool attribute_ignored_p (const attribute_spec *const);
     44 
     45 extern struct scoped_attributes* register_scoped_attributes (const struct attribute_spec *,
     46 							     const char *,
     47 							     bool = false);
     48 
     49 extern char *sorted_attr_string (tree);
     50 extern bool common_function_versions (tree, tree);
     51 extern tree make_dispatcher_decl (const tree);
     52 extern bool is_function_default_version (const tree);
     53 extern void handle_ignored_attributes_option (vec<char *> *);
     54 
     55 /* Return a type like TTYPE except that its TYPE_ATTRIBUTES
     56    is ATTRIBUTE.
     57 
     58    Such modified types already made are recorded so that duplicates
     59    are not made.  */
     60 
     61 extern tree build_type_attribute_variant (tree, tree);
     62 extern tree build_decl_attribute_variant (tree, tree);
     63 extern tree build_type_attribute_qual_variant (tree, tree, int);
     64 
     65 extern bool simple_cst_list_equal (const_tree, const_tree);
     66 extern bool attribute_value_equal (const_tree, const_tree);
     67 
     68 /* Return 0 if the attributes for two types are incompatible, 1 if they
     69    are compatible, and 2 if they are nearly compatible (which causes a
     70    warning to be generated).  */
     71 extern int comp_type_attributes (const_tree, const_tree);
     72 
     73 extern tree affects_type_identity_attributes (tree, bool = true);
     74 extern tree restrict_type_identity_attributes_to (tree, tree);
     75 
     76 /* Default versions of target-overridable functions.  */
     77 extern tree merge_decl_attributes (tree, tree);
     78 extern tree merge_type_attributes (tree, tree);
     79 
     80 /* Remove any instances of attribute ATTR_NAME in LIST and return the
     81    modified list.  */
     82 
     83 extern tree remove_attribute (const char *, tree);
     84 
     85 /* Given two attributes lists, return a list of their union.  */
     86 
     87 extern tree merge_attributes (tree, tree);
     88 
     89 /* Duplicate all attributes with name NAME in ATTR list to *ATTRS if
     90    they are missing there.  */
     91 
     92 extern void duplicate_one_attribute (tree *, tree, const char *);
     93 
     94 /* Duplicate all attributes from user DECL to the corresponding
     95    builtin that should be propagated.  */
     96 
     97 extern void copy_attributes_to_builtin (tree);
     98 
     99 /* Given two Windows decl attributes lists, possibly including
    100    dllimport, return a list of their union .  */
    101 extern tree merge_dllimport_decl_attributes (tree, tree);
    102 
    103 /* Handle a "dllimport" or "dllexport" attribute.  */
    104 extern tree handle_dll_attribute (tree *, tree, tree, int, bool *);
    105 
    106 extern int attribute_list_equal (const_tree, const_tree);
    107 extern int attribute_list_contained (const_tree, const_tree);
    108 
    109 /* The backbone of lookup_attribute().  ATTR_LEN is the string length
    110    of ATTR_NAME, and LIST is not NULL_TREE.
    111 
    112    The function is called from lookup_attribute in order to optimize
    113    for size.  */
    114 extern tree private_lookup_attribute (const char *attr_name, size_t attr_len,
    115 				      tree list);
    116 
    117 extern unsigned decls_mismatched_attributes (tree, tree, tree,
    118 					     const char* const[],
    119 					     pretty_printer*);
    120 
    121 extern void maybe_diag_alias_attributes (tree, tree);
    122 
    123 /* For a given string S of length L, strip leading and trailing '_' characters
    124    so that we have a canonical form of attribute names.  NB: This function may
    125    change S and L.  */
    126 
    127 template <typename T>
    128 inline bool
    129 canonicalize_attr_name (const char *&s, T &l)
    130 {
    131   if (l > 4 && s[0] == '_' && s[1] == '_' && s[l - 1] == '_' && s[l - 2] == '_')
    132     {
    133       s += 2;
    134       l -= 4;
    135       return true;
    136     }
    137   return false;
    138 }
    139 
    140 /* For a given IDENTIFIER_NODE, strip leading and trailing '_' characters
    141    so that we have a canonical form of attribute names.  */
    142 
    143 static inline tree
    144 canonicalize_attr_name (tree attr_name)
    145 {
    146   size_t l = IDENTIFIER_LENGTH (attr_name);
    147   const char *s = IDENTIFIER_POINTER (attr_name);
    148 
    149   if (canonicalize_attr_name (s, l))
    150     return get_identifier_with_length (s, l);
    151 
    152   return attr_name;
    153 }
    154 
    155 /* Compare attribute identifiers ATTR1 and ATTR2 with length ATTR1_LEN and
    156    ATTR2_LEN.  */
    157 
    158 static inline bool
    159 cmp_attribs (const char *attr1, size_t attr1_len,
    160 	     const char *attr2, size_t attr2_len)
    161 {
    162   return attr1_len == attr2_len && strncmp (attr1, attr2, attr1_len) == 0;
    163 }
    164 
    165 /* Compare attribute identifiers ATTR1 and ATTR2.  */
    166 
    167 static inline bool
    168 cmp_attribs (const char *attr1, const char *attr2)
    169 {
    170   return cmp_attribs (attr1, strlen (attr1), attr2, strlen (attr2));
    171 }
    172 
    173 /* Given an identifier node IDENT and a string ATTR_NAME, return true
    174    if the identifier node is a valid attribute name for the string.  */
    175 
    176 static inline bool
    177 is_attribute_p (const char *attr_name, const_tree ident)
    178 {
    179   return cmp_attribs (attr_name, strlen (attr_name),
    180 		      IDENTIFIER_POINTER (ident), IDENTIFIER_LENGTH (ident));
    181 }
    182 
    183 /* Given an attribute name ATTR_NAME and a list of attributes LIST,
    184    return a pointer to the attribute's list element if the attribute
    185    is part of the list, or NULL_TREE if not found.  If the attribute
    186    appears more than once, this only returns the first occurrence; the
    187    TREE_CHAIN of the return value should be passed back in if further
    188    occurrences are wanted.  ATTR_NAME must be in the form 'text' (not
    189    '__text__').  */
    190 
    191 static inline tree
    192 lookup_attribute (const char *attr_name, tree list)
    193 {
    194   if (CHECKING_P && attr_name[0] != '_')
    195     {
    196       size_t attr_len = strlen (attr_name);
    197       gcc_checking_assert (!canonicalize_attr_name (attr_name, attr_len));
    198     }
    199   /* In most cases, list is NULL_TREE.  */
    200   if (list == NULL_TREE)
    201     return NULL_TREE;
    202   else
    203     {
    204       size_t attr_len = strlen (attr_name);
    205       /* Do the strlen() before calling the out-of-line implementation.
    206 	 In most cases attr_name is a string constant, and the compiler
    207 	 will optimize the strlen() away.  */
    208       return private_lookup_attribute (attr_name, attr_len, list);
    209     }
    210 }
    211 
    212 /* Given an attribute name ATTR_NAME and a list of attributes LIST,
    213    return a pointer to the attribute's list first element if the attribute
    214    starts with ATTR_NAME.  ATTR_NAME must be in the form 'text' (not
    215    '__text__').  */
    216 
    217 static inline tree
    218 lookup_attribute_by_prefix (const char *attr_name, tree list)
    219 {
    220   gcc_checking_assert (attr_name[0] != '_');
    221   /* In most cases, list is NULL_TREE.  */
    222   if (list == NULL_TREE)
    223     return NULL_TREE;
    224   else
    225     {
    226       size_t attr_len = strlen (attr_name);
    227       while (list)
    228 	{
    229 	  tree name = get_attribute_name (list);
    230 	  size_t ident_len = IDENTIFIER_LENGTH (name);
    231 
    232 	  if (attr_len > ident_len)
    233 	    {
    234 	      list = TREE_CHAIN (list);
    235 	      continue;
    236 	    }
    237 
    238 	  const char *p = IDENTIFIER_POINTER (name);
    239 	  gcc_checking_assert (attr_len == 0 || p[0] != '_');
    240 
    241 	  if (strncmp (attr_name, p, attr_len) == 0)
    242 	    break;
    243 
    244 	  list = TREE_CHAIN (list);
    245 	}
    246 
    247       return list;
    248     }
    249 }
    250 
    251 /* Description of a function argument declared with attribute access.
    252    Used as an "iterator" over all such arguments in a function declaration
    253    or call.  */
    254 
    255 struct attr_access
    256 {
    257   /* The beginning and end of the internal string representation.  */
    258   const char *str, *end;
    259   /* The attribute pointer argument.  */
    260   tree ptr;
    261   /* For a declaration, a TREE_CHAIN of VLA bound expressions stored
    262      in TREE_VALUE and their positions in the argument list (stored
    263      in TREE_PURPOSE).  Each expression may be a PARM_DECL or some
    264      other DECL (for ordinary variables), or an EXPR for other
    265      expressions (e.g., funcion calls).  */
    266   tree size;
    267 
    268   /* The zero-based position of each of the formal function arguments.
    269      For the optional SIZARG, UINT_MAX when not specified.  For VLAs
    270      with multiple variable bounds, SIZARG is the position corresponding
    271      to the most significant bound in the argument list.  Positions of
    272      subsequent bounds are in the TREE_PURPOSE field of the SIZE chain.  */
    273   unsigned ptrarg;
    274   unsigned sizarg;
    275   /* For internal specifications only, the constant minimum size of
    276      the array, zero if not specified, and HWI_M1U for the unspecified
    277      VLA [*] notation.  Meaningless for external (explicit) access
    278      specifications.  */
    279   unsigned HOST_WIDE_INT minsize;
    280 
    281   /* The access mode.  */
    282   access_mode mode;
    283 
    284   /* Set for an attribute added internally rather than by an explicit
    285      declaration. */
    286   bool internal_p;
    287   /* Set for the T[static MINSIZE] array notation for nonzero MINSIZE
    288      less than HWI_M1U.  */
    289   bool static_p;
    290 
    291   /* Return the number of specified VLA bounds.  */
    292   unsigned vla_bounds (unsigned *) const;
    293 
    294   /* Return internal representation as STRING_CST.  */
    295   tree to_internal_string () const;
    296 
    297   /* Return the human-readable representation of the external attribute
    298      specification (as it might appear in the source code) as STRING_CST.  */
    299   tree to_external_string () const;
    300 
    301   /* Return argument of array type formatted as a readable string.  */
    302   std::string array_as_string (tree) const;
    303 
    304   /* Return the access mode corresponding to the character code.  */
    305   static access_mode from_mode_char (char);
    306 
    307   /* Reset front end-specific attribute access data from attributes.  */
    308   static void free_lang_data (tree);
    309 
    310   /* The character codes corresponding to all the access modes.  */
    311   static constexpr char mode_chars[5] = { '-', 'r', 'w', 'x', '^' };
    312 
    313   /* The strings corresponding to just the external access modes.  */
    314   static constexpr char mode_names[4][11] =
    315     {
    316      "none", "read_only", "write_only", "read_write"
    317     };
    318 };
    319 
    320 inline access_mode
    321 attr_access::from_mode_char (char c)
    322 {
    323   switch (c)
    324     {
    325     case mode_chars[access_none]: return access_none;
    326     case mode_chars[access_read_only]: return access_read_only;
    327     case mode_chars[access_write_only]: return access_write_only;
    328     case mode_chars[access_read_write]: return access_read_write;
    329     case mode_chars[access_deferred]: return access_deferred;
    330     }
    331   gcc_unreachable ();
    332 }
    333 
    334 /* Used to define rdwr_map below.  */
    335 struct rdwr_access_hash: int_hash<int, -1> { };
    336 
    337 /* A mapping between argument number corresponding to attribute access
    338    mode (read_only, write_only, or read_write) and operands.  */
    339 struct attr_access;
    340 typedef hash_map<rdwr_access_hash, attr_access> rdwr_map;
    341 
    342 extern void init_attr_rdwr_indices (rdwr_map *, tree);
    343 extern attr_access *get_parm_access (rdwr_map &, tree,
    344 				     tree = current_function_decl);
    345 
    346 #endif // GCC_ATTRIBS_H
    347