Studying note of GCC-3.4.6 source (60)

4.3.1.7.5.6.5.              Processing attributes of builtin

4.3.1.7.5.6.5.1.        Install attribute handlers

Though we create nodes for attributes of builtin functions, we still don’t install them into the builtin function. And even there is no attribute, we should express this fact in the function node explicitly.

 

builtin_function_1 (continue)

 

3253     /* Possibly apply some default attributes to this built-in function.  */

3254     if (attrs)

3255       decl_attributes (&decl, attrs, ATTR_FLAG_BUILT_IN);

3256     else

3257       decl_attributes (&decl, NULL_TREE, 0);

3258  

3259     return decl;

3260   }

 

Obviously, the language determines the set of available attributes, further some attributes’ processing methods depend on target. So first, we need to select out the correct methods according to the language and the target.

 

139    tree

140    decl_attributes (tree *node, tree attributes, int flags)                                          in attribs.c

141    {

142      tree a;

143      tree returned_attrs = NULL_TREE;

144   

145      if (!attributes_initialized)

146        init_attributes ();

 

Attribute handling methods are collected by attribute_spec with other running information. In GCC v3.4.6, the methods are grouped into 4 sets.

 

42      static const struct attribute_spec *attribute_tables[4];                                   in attribs.c

 

2181   struct attribute_spec                                                                                 in tree.h

2182   {

2183     /* The name of the attribute (without any leading or trailing __),

2184       or NULL to mark the end of a table of attributes.  */

2185     const char *const name;

2186     /* The minimum length of the list of arguments of the attribute.  */

2187     const int min_length;

2188     /* The maximum length of the list of arguments of the attribute

2189       (-1 for no maximum).  */

2190     const int max_length;

2191     /* Whether this attribute requires a DECL. If it does, it will be passed

2192       from types of DECLs, function return types and array element types to

2193       the DECLs, function types and array types respectively; but when

2194       applied to a type in any other circumstances, it will be ignored with

2195       a warning. (If greater control is desired for a given attribute,

2196       this should be false, and the flags argument to the handler may be

2197       used to gain greater control in that case.)  */

2198     const bool decl_required;

2199     /* Whether this attribute requires a type. If it does, it will be passed

2200       from a DECL to the type of that DECL.  */

2201     const bool type_required;

2202    /* Whether this attribute requires a function (or method) type. If it does,

2203       it will be passed from a function pointer type to the target type,

2204       and from a function return type (which is not itself a function

2205       pointer type) to the function type.  */

2206     const bool function_type_required;

2207     /* Function to handle this attribute. NODE points to the node to which

2208       the attribute is to be applied. If a DECL, it should be modified in

2209       place; if a TYPE, a copy should be created. NAME is the name of the

2210       attribute (possibly with leading or trailing __). ARGS is the TREE_LIST

2211       of the arguments (which may be NULL). FLAGS gives further information

2212       about the context of the attribute. Afterwards, the attributes will

2213       be added to the DECL_ATTRIBUTES or TYPE_ATTRIBUTES, as appropriate,

2214       unless *NO_ADD_ATTRS is set to true (which should be done on error,

2215       as well as in any other cases when the attributes should not be added

2216       to the DECL or TYPE). Depending on FLAGS, any attributes to be

2217       applied to another type or DECL later may be returned;

2218       otherwise the return value should be NULL_TREE. This pointer may be

2219       NULL if no special handling is required beyond the checks implied

2220       by the rest of this structure.  */

2221     tree (*const handler) (tree *node, tree name, tree args,

2222                             int flags, bool *no_add_attrs);

2223   };

 

The detail information about these 4 sets are given in following 4 tables.

name

min args

max args

del_req

type_req

func_req

handler

packed

0

0

false

false

false

handle_packed_attribute

nocommon

0

0

true

false

false

handle_nocommon_attribute

common

0

0

true

false

false

handle_common_attribute

noreturn

0

0

true

false

false

handle_noreturn_attribute

volatile

0

0

true

false

false

handle_noreturn_attribute

noinline

0

0

true

false

false

handle_noinline_attribute

always_inline

0

0

true

false

false

handle_always_inline_attribute

used

0

0

true

false

false

handle_used_attribute

unused

0

0

false

false

false

handle_unused_attribute

const

0

0

true

false

false

handle_const_attribute

transparent_union

0

0

false

false

false

handle_transparent_union_attribute

constructor

0

0

true

false

false

handle_constructor_attribute

destructor

0

0

true

false

false

handle_destructor_attribute

mode

1

1

false

true

false

handle_mode_attribute

section

1

1

true

false

false

handle_section_attribute

aligned

0

1

false

false

false

handle_aligned_attribute

weak

0

0

true

false

false

handle_weak_attribute

alias

1

1

true

false

false

handle_alias_attribute

no_instrument_function

0

0

true

false

false

handle_no_instrument_function_attribute

malloc

0

0

true

false

false

handle_malloc_attribute

no_stack_limit

0

0

true

false

false

handle_no_limit_stack_attribute

pure

0

0

true

false

false

handle_pure_attribute

deprecated

0

0

false

false

false

handle_deprecated_attribute

vector_size

1

1

false

true

false

handle_vector_size_attribute

visibility

1

1

true

false

false

handle_visibility_attribute

tls_model

1

1

true

false

false

handle_tls_model_attribute

nonnull

0

-1

false

true

true

handle_nonnull_attribute

nothrow

0

0

true

false

false

handle_nothrow_attribute

may_alias

0

0

false

true

false

NULL

cleanup

1

1

true

false

false

handle_cleanup_attribute

warn_unused_result

0

0

false

true

true

handle_warn_unused_result_attribute

Table 12: builtin functions of common_attribute_table for C++

name

min args

max args

del_req

type_req

func_req

handler

java_interface

0

0

false

false

false

handle_java_interface_attribute

com_interface

0

0

false

false

false

handle_com_interface_attribute

init_priority

1

1

true

false

false

handle_init_priority_attribute

Table 13: builtin function of attribute_table for C++

name

min args

max args

del_req

type_req

func_req

handler

format

3

3

false

true

true

handle_format_attribute

format_arg

1

1

false

true

true

handle_format_arg_attribute

Table 14: builtin function of format_attribute_table for C++

name

min args

max args

del_req

type_req

func_req

handler

stdcall

0

0

false

true

true

ix86_handle_cdecl_attribute

fastcall

0

0

false

true

true

ix86_handle_cdecl_attribute

cdecl

0

0

false

true

true

ix86_handle_cdecl_attribute

regparm

1

1

false

true

true

ix86_handle_regparm_attribute

ms_struct

0

0

false

false

false

ix86_handle_struct_attribute

gcc_struct

0

0

false

false

false

ix86_handle_struct_attribute

Table 15: builtin function of ix86_attribute_table for x86 machine

In below code, it can see that the first 3 tables are set by language hook, which have their content filled by the language; and in the last table are attributes unrelated to language but depend on target.

 

55      static void

56      init_attributes (void)                                                                                in attribs.c

57      {

58        size_t i;

59     

60        attribute_tables[0] = lang_hooks.common_attribute_table;

61        attribute_tables[1] = lang_hooks.attribute_table;

62        attribute_tables[2] = lang_hooks.format_attribute_table;

63        attribute_tables[3] = targetm.attribute_table;

64     

65        /* Translate NULL pointers to pointers to the empty table.  */

66        for (i = 0; i < ARRAY_SIZE (attribute_tables); i++)

67          if (attribute_tables[i] == NULL)

68            attribute_tables[i] = empty_attribute_table;

69     

70      #ifdef ENABLE_CHECKING

       ...

125    #endif

126   

127      attributes_initialized = true;

128    }

 

Remember that in the builtin function “__builtin_vprintf”, its attributes are given in built_in_attributes [ATTR_FORMAT_PRINTF_1_0], and its structure is shown is below figure.

In this image, it is clearly that attributes are linked tegother by field chain, with field purpose tells out what it is for and field value gives out its content. Notice that purpose field in fact is a node of identifier.

 

decl_attributes (continue)

 

148      (*targetm.insert_attributes) (*node, &attributes);

149   

150      for (a = attributes; a; a = TREE_CHAIN (a))

151      {

152        tree name = TREE_PURPOSE (a);

153        tree args = TREE_VALUE (a);

154        tree *anode = node;

155        const struct attribute_spec *spec = NULL;

156        bool no_add_attrs = 0;

157        tree fn_ptr_tmp = NULL_TREE;

158        size_t i;

159   

160        for (i = 0; i < ARRAY_SIZE (attribute_tables); i++)

161        {

162          int j;

163   

164          for (j = 0; attribute_tables[i][j].name != NULL; j++)

165          {

166            if (is_attribute_p (attribute_tables[i][j].name, name))

167            {

168              spec = &attribute_tables[i][j];

169              break;

170            }

171          }

172          if (spec != NULL)

173            break;

174        }

175   

176        if (spec == NULL)

177        {

178          warning ("`%s' attribute directive ignored",

179                  IDENTIFIER_POINTER (name));

180          continue;

181        }

182        else if (list_length (args) < spec->min_length

183              || (spec->max_length >= 0

184                && list_length (args) > spec->max_length))

185        {

186          error ("wrong number of arguments specified for `%s' attribute",

187                IDENTIFIER_POINTER (name));

188          continue;

189        }

 

At line 148, for x86 machine, insert_attributes is not redefined. It does nothing by default. Andnote that anode and node are both pointer of pointer, node always keeps the address of the tree node, and anode transfer within the node for searching the real object the attributes being applied. To clarify the real kind of the node, argument flags is used give out the detail. Currently, below enumerators are in used:

ATTR_FLAG_DECL_NEXT: node is the type of a declaration, and any attributes that should be passed in again to be applied to the declaration rather than the type should be returned.

ATTR_FLAG_FUNCTION_NEXT: node is a function-return-type, and any attributes that should be passed in again to be applied to the function type rather than the return type should be returned.

ATTR_FLAG_ARRAY_NEXT: node is an array-element-type, and any attributes that should be passed in again to be applied to the array type rather than the element type should be returned.

ATTR_FLAG_TYPE_IN_PLACE: node is a struct, union or enum type being created, and should be modified in place.

ATTR_FLAG_BUILT_IN: The attributes are being applied by default to a library function whose name indicates known behavior, and should be silently ignored if they are not in fact compatible with the function type.

Here for builtin “__builtin_vprintf”, argument flags is ATTR_FLAG_BUILT_IN.

 

decl_attributes (continue)

 

191        if (spec->decl_required && !DECL_P (*anode))

192        {

193          if (flags & ((int) ATTR_FLAG_DECL_NEXT

194                       | (int) ATTR_FLAG_FUNCTION_NEXT

195                       | (int) ATTR_FLAG_ARRAY_NEXT))

196          {

197            /* Pass on this attribute to be tried again.  */

198            returned_attrs = tree_cons (name, args, returned_attrs);

199            continue;

200          }

201          else

202          {

203            warning ("`%s' attribute does not apply to types",

204                    IDENTIFIER_POINTER (name));

205            continue;

206          }

207        }

208   

209        /* If we require a type, but were passed a decl, set up to make a

210          new type and update the one in the decl. ATTR_FLAG_TYPE_IN_PLACE

211           would have applied if we'd been passed a type, but we cannot modify

212          the decl's type in place here.  */

213        if (spec->type_required && DECL_P (*anode))

214        {

215          anode = &TREE_TYPE (*anode);

216          flags &= ~(int) ATTR_FLAG_TYPE_IN_PLACE;

217        }

218   

219        if (spec->function_type_required && TREE_CODE (*anode) != FUNCTION_TYPE

220            && TREE_CODE (*anode) != METHOD_TYPE)

221        {

222          if (TREE_CODE (*anode) == POINTER_TYPE

223             && (TREE_CODE (TREE_TYPE (*anode)) == FUNCTION_TYPE

224                  || TREE_CODE (TREE_TYPE (*anode)) == METHOD_TYPE))

225          {

226            /* OK, this is a bit convoluted. We can't just make a copy

227              of the pointer type and modify its TREE_TYPE, because if

228              we change the attributes of the target type the pointer

229              type needs to have a different TYPE_MAIN_VARIANT. So we

230              pull out the target type now, frob it as appropriate, and

231              rebuild the pointer type later.

232   

233              This would all be simpler if attributes were part of the

234              declarator, grumble grumble.  */

235            fn_ptr_tmp = TREE_TYPE (*anode);

236            anode = &fn_ptr_tmp;

237            flags &= ~(int) ATTR_FLAG_TYPE_IN_PLACE;

238          }

239          else if (flags & (int) ATTR_FLAG_FUNCTION_NEXT)

240          {

241            /* Pass on this attribute to be tried again.  */

242            returned_attrs = tree_cons (name, args, returned_attrs);

243            continue;

244          }

245   

246          if (TREE_CODE (*anode) != FUNCTION_TYPE

247               && TREE_CODE (*anode) != METHOD_TYPE)

248          {

249            warning ("`%s' attribute only applies to function types",

250                       IDENTIFIER_POINTER (name));

251            continue;

252          }

253        }

254   

255        if (spec->handler != NULL)

256          returned_attrs = chainon ((*spec->handler) (anode, name, args,

257                                             flags, &no_add_attrs),

258                               returned_attrs);

 

Refer to builtin function for format_attribute_table for C++, for attribute of format, it requires type and function type, but node is a FUNCTION_DECL. However, at line 215, anode is updated to the type of declaration - builtin_types [BT_FN_INT_CONST_STRING_VALIST_ARG].

 

你可能感兴趣的:(function,struct,tree,table,attributes,Types)