]> git.ipfire.org Git - thirdparty/gcc.git/blame - gcc/config/darwin.c
2009-09-24 H.J. Lu <hongjiu.lu@intel.com>
[thirdparty/gcc.git] / gcc / config / darwin.c
CommitLineData
80d725d7 1/* Functions for generic Darwin as target machine for GNU C compiler.
2be1b07e 2 Copyright (C) 1989, 1990, 1991, 1992, 1993, 2000, 2001, 2002, 2003, 2004,
67a779df 3 2005, 2006, 2007, 2008, 2009
80d725d7 4 Free Software Foundation, Inc.
5 Contributed by Apple Computer Inc.
6
187b36cf 7This file is part of GCC.
80d725d7 8
187b36cf 9GCC is free software; you can redistribute it and/or modify
80d725d7 10it under the terms of the GNU General Public License as published by
038d1e19 11the Free Software Foundation; either version 3, or (at your option)
80d725d7 12any later version.
13
187b36cf 14GCC is distributed in the hope that it will be useful,
80d725d7 15but WITHOUT ANY WARRANTY; without even the implied warranty of
16MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17GNU General Public License for more details.
18
19You should have received a copy of the GNU General Public License
038d1e19 20along with GCC; see the file COPYING3. If not see
21<http://www.gnu.org/licenses/>. */
80d725d7 22
23#include "config.h"
24#include "system.h"
805e22b2 25#include "coretypes.h"
26#include "tm.h"
80d725d7 27#include "rtl.h"
28#include "regs.h"
29#include "hard-reg-set.h"
30#include "real.h"
31#include "insn-config.h"
32#include "conditions.h"
33#include "insn-flags.h"
34#include "output.h"
35#include "insn-attr.h"
36#include "flags.h"
37#include "tree.h"
38#include "expr.h"
39#include "reload.h"
80d725d7 40#include "function.h"
41#include "ggc.h"
d19bd1f0 42#include "langhooks.h"
6d3abfc0 43#include "target.h"
79e2a0ca 44#include "tm_p.h"
690abe5d 45#include "toplev.h"
ab06d2ff 46#include "hashtab.h"
3072d30e 47#include "df.h"
cf756914 48#include "debug.h"
80d725d7 49
f0ed0e8a 50/* Darwin supports a feature called fix-and-continue, which is used
51 for rapid turn around debugging. When code is compiled with the
52 -mfix-and-continue flag, two changes are made to the generated code
53 that allow the system to do things that it would normally not be
54 able to do easily. These changes allow gdb to load in
55 recompilation of a translation unit that has been changed into a
56 running program and replace existing functions and methods of that
cb4213cd 57 translation unit with versions of those functions and methods
f0ed0e8a 58 from the newly compiled translation unit. The new functions access
d188633a 59 the existing static symbols from the old translation unit, if the
60 symbol existed in the unit to be replaced, and from the new
61 translation unit, otherwise.
f0ed0e8a 62
8063e558 63 The changes are to insert 5 nops at the beginning of all functions
d188633a 64 and to use indirection to get at static symbols. The 5 nops
f0ed0e8a 65 are required by consumers of the generated code. Currently, gdb
66 uses this to patch in a jump to the overriding function, this
67 allows all uses of the old name to forward to the replacement,
84cbcde5 68 including existing function pointers and virtual methods. See
f0ed0e8a 69 rs6000_emit_prologue for the code that handles the nop insertions.
b215c058 70
f0ed0e8a 71 The added indirection allows gdb to redirect accesses to static
d188633a 72 symbols from the newly loaded translation unit to the existing
73 symbol, if any. @code{static} symbols are special and are handled by
74 setting the second word in the .non_lazy_symbol_pointer data
75 structure to symbol. See indirect_data for the code that handles
76 the extra indirection, and machopic_output_indirection and its use
77 of MACHO_SYMBOL_STATIC for the code that handles @code{static}
78 symbol indirection. */
f0ed0e8a 79
92e4ba7d 80/* Section names. */
81section * darwin_sections[NUM_DARWIN_SECTIONS];
2f14b1f9 82
f720b7c6 83/* True if we're setting __attribute__ ((ms_struct)). */
84int darwin_ms_struct = false;
85
2f14b1f9 86/* A get_unnamed_section callback used to switch to an ObjC section.
87 DIRECTIVE is as for output_section_asm_op. */
88
89static void
90output_objc_section_asm_op (const void *directive)
91{
92e4ba7d 92 static bool been_here = false;
2f14b1f9 93
92e4ba7d 94 if (! been_here)
2f14b1f9 95 {
b215c058 96 static const enum darwin_section_enum tomark[] =
92e4ba7d 97 {
98 /* written, cold -> hot */
99 objc_cat_cls_meth_section,
100 objc_cat_inst_meth_section,
101 objc_string_object_section,
102 objc_constant_string_object_section,
103 objc_selector_refs_section,
104 objc_selector_fixup_section,
105 objc_cls_refs_section,
106 objc_class_section,
107 objc_meta_class_section,
108 /* shared, hot -> cold */
109 objc_cls_meth_section,
110 objc_inst_meth_section,
111 objc_protocol_section,
112 objc_class_names_section,
113 objc_meth_var_types_section,
114 objc_meth_var_names_section,
115 objc_category_section,
116 objc_class_vars_section,
117 objc_instance_vars_section,
118 objc_module_info_section,
119 objc_symbols_section
120 };
121 size_t i;
b215c058 122
92e4ba7d 123 been_here = true;
124 for (i = 0; i < ARRAY_SIZE (tomark); i++)
125 switch_to_section (darwin_sections[tomark[i]]);
2f14b1f9 126 }
127 output_section_asm_op (directive);
128}
129
130/* Implement TARGET_ASM_INIT_SECTIONS. */
131
132void
133darwin_init_sections (void)
134{
92e4ba7d 135#define DEF_SECTION(NAME, FLAGS, DIRECTIVE, OBJC) \
136 darwin_sections[NAME] = \
137 get_unnamed_section (FLAGS, (OBJC \
138 ? output_objc_section_asm_op \
139 : output_section_asm_op), \
140 "\t" DIRECTIVE);
141#include "config/darwin-sections.def"
2f14b1f9 142#undef DEF_SECTION
143
92e4ba7d 144 readonly_data_section = darwin_sections[const_section];
145 exception_section = darwin_sections[darwin_exception_section];
146 eh_frame_section = darwin_sections[darwin_eh_frame_section];
2f14b1f9 147}
f0ed0e8a 148
80d725d7 149int
b40da9a7 150name_needs_quotes (const char *name)
80d725d7 151{
152 int c;
153 while ((c = *name++) != '\0')
37ec5024 154 if (! ISIDNUM (c) && c != '.' && c != '$')
80d725d7 155 return 1;
156 return 0;
157}
158
866318b4 159/* Return true if SYM_REF can be used without an indirection. */
ab06d2ff 160static int
161machopic_symbol_defined_p (rtx sym_ref)
80d725d7 162{
866318b4 163 if (SYMBOL_REF_FLAGS (sym_ref) & MACHO_SYMBOL_FLAG_DEFINED)
164 return true;
165
166 /* If a symbol references local and is not an extern to this
167 file, then the symbol might be able to declared as defined. */
168 if (SYMBOL_REF_LOCAL_P (sym_ref) && ! SYMBOL_REF_EXTERNAL_P (sym_ref))
169 {
170 /* If the symbol references a variable and the variable is a
171 common symbol, then this symbol is not defined. */
172 if (SYMBOL_REF_FLAGS (sym_ref) & MACHO_SYMBOL_FLAG_VARIABLE)
173 {
174 tree decl = SYMBOL_REF_DECL (sym_ref);
175 if (!decl)
176 return true;
177 if (DECL_COMMON (decl))
178 return false;
179 }
180 return true;
181 }
182 return false;
80d725d7 183}
184
ab06d2ff 185/* This module assumes that (const (symbol_ref "foo")) is a legal pic
186 reference, which will not be changed. */
b40da9a7 187
80d725d7 188enum machopic_addr_class
ab06d2ff 189machopic_classify_symbol (rtx sym_ref)
80d725d7 190{
ab06d2ff 191 int flags;
192 bool function_p;
193
194 flags = SYMBOL_REF_FLAGS (sym_ref);
195 function_p = SYMBOL_REF_FUNCTION_P (sym_ref);
196 if (machopic_symbol_defined_p (sym_ref))
b215c058 197 return (function_p
ab06d2ff 198 ? MACHOPIC_DEFINED_FUNCTION : MACHOPIC_DEFINED_DATA);
199 else
b215c058 200 return (function_p
ab06d2ff 201 ? MACHOPIC_UNDEFINED_FUNCTION : MACHOPIC_UNDEFINED_DATA);
80d725d7 202}
203
f0ed0e8a 204#ifndef TARGET_FIX_AND_CONTINUE
205#define TARGET_FIX_AND_CONTINUE 0
206#endif
207
208/* Indicate when fix-and-continue style code generation is being used
209 and when a reference to data should be indirected so that it can be
0975351b 210 rebound in a new translation unit to reference the original instance
f0ed0e8a 211 of that data. Symbol names that are for code generation local to
212 the translation unit are bound to the new translation unit;
213 currently this means symbols that begin with L or _OBJC_;
214 otherwise, we indicate that an indirect reference should be made to
215 permit the runtime to rebind new instances of the translation unit
216 to the original instance of the data. */
217
218static int
219indirect_data (rtx sym_ref)
220{
221 int lprefix;
222 const char *name;
223
c4f8e01f 224 /* If we aren't generating fix-and-continue code, don't do anything
225 special. */
f0ed0e8a 226 if (TARGET_FIX_AND_CONTINUE == 0)
227 return 0;
228
229 /* Otherwise, all symbol except symbols that begin with L or _OBJC_
230 are indirected. Symbols that begin with L and _OBJC_ are always
231 bound to the current translation unit as they are used for
232 generated local data of the translation unit. */
233
234 name = XSTR (sym_ref, 0);
235
236 lprefix = (((name[0] == '*' || name[0] == '&')
237 && (name[1] == 'L' || (name[1] == '"' && name[2] == 'L')))
78a79402 238 || (strncmp (name, "_OBJC_", 6) == 0));
f0ed0e8a 239
240 return ! lprefix;
241}
242
243
80d725d7 244static int
ab06d2ff 245machopic_data_defined_p (rtx sym_ref)
80d725d7 246{
f0ed0e8a 247 if (indirect_data (sym_ref))
248 return 0;
249
ab06d2ff 250 switch (machopic_classify_symbol (sym_ref))
80d725d7 251 {
252 case MACHOPIC_DEFINED_DATA:
3a7ccce8 253 case MACHOPIC_DEFINED_FUNCTION:
80d725d7 254 return 1;
255 default:
256 return 0;
257 }
258}
259
80d725d7 260void
ab06d2ff 261machopic_define_symbol (rtx mem)
80d725d7 262{
ab06d2ff 263 rtx sym_ref;
b215c058 264
27d2baef 265 gcc_assert (GET_CODE (mem) == MEM);
ab06d2ff 266 sym_ref = XEXP (mem, 0);
267 SYMBOL_REF_FLAGS (sym_ref) |= MACHO_SYMBOL_FLAG_DEFINED;
80d725d7 268}
269
a914e728 270/* Return either ORIG or:
80d725d7 271
a914e728 272 (const:P (unspec:P [ORIG] UNSPEC_MACHOPIC_OFFSET))
ab06d2ff 273
a914e728 274 depending on MACHO_DYNAMIC_NO_PIC_P. */
ab06d2ff 275rtx
a914e728 276machopic_gen_offset (rtx orig)
ab06d2ff 277{
a914e728 278 if (MACHO_DYNAMIC_NO_PIC_P)
b06a3cc3 279 return orig;
280 else
a914e728 281 {
282 /* Play games to avoid marking the function as needing pic if we
283 are being called as part of the cost-estimation process. */
e5d1687d 284 if (current_ir_type () != IR_GIMPLE || currently_expanding_to_rtl)
a914e728 285 crtl->uses_pic_offset_table = 1;
286 orig = gen_rtx_UNSPEC (Pmode, gen_rtvec (1, orig),
287 UNSPEC_MACHOPIC_OFFSET);
288 return gen_rtx_CONST (Pmode, orig);
289 }
b06a3cc3 290}
291
b186bdc9 292static GTY(()) const char * function_base_func_name;
293static GTY(()) int current_pic_label_num;
294
295void
296machopic_output_function_base_name (FILE *file)
297{
298 const char *current_name;
299
1d60d981 300 /* If dynamic-no-pic is on, we should not get here. */
27d2baef 301 gcc_assert (!MACHO_DYNAMIC_NO_PIC_P);
b40da9a7 302 current_name =
b186bdc9 303 IDENTIFIER_POINTER (DECL_ASSEMBLER_NAME (current_function_decl));
3a32429b 304 if (function_base_func_name != current_name)
80d725d7 305 {
80d725d7 306 ++current_pic_label_num;
3a32429b 307 function_base_func_name = current_name;
80d725d7 308 }
b186bdc9 309 fprintf (file, "\"L%011d$pb\"", current_pic_label_num);
80d725d7 310}
311
ab06d2ff 312/* The suffix attached to non-lazy pointer symbols. */
313#define NON_LAZY_POINTER_SUFFIX "$non_lazy_ptr"
314/* The suffix attached to stub symbols. */
315#define STUB_SUFFIX "$stub"
80d725d7 316
fb1e4f4a 317typedef struct GTY (()) machopic_indirection
80d725d7 318{
ab06d2ff 319 /* The SYMBOL_REF for the entity referenced. */
320 rtx symbol;
6d3abfc0 321 /* The name of the stub or non-lazy pointer. */
322 const char * ptr_name;
ab06d2ff 323 /* True iff this entry is for a stub (as opposed to a non-lazy
324 pointer). */
325 bool stub_p;
326 /* True iff this stub or pointer pointer has been referenced. */
327 bool used;
328} machopic_indirection;
329
330/* A table mapping stub names and non-lazy pointer names to
331 SYMBOL_REFs for the stubbed-to and pointed-to entities. */
332
b215c058 333static GTY ((param_is (struct machopic_indirection))) htab_t
ab06d2ff 334 machopic_indirections;
335
336/* Return a hash value for a SLOT in the indirections hash table. */
337
338static hashval_t
339machopic_indirection_hash (const void *slot)
340{
341 const machopic_indirection *p = (const machopic_indirection *) slot;
6d3abfc0 342 return htab_hash_string (p->ptr_name);
ab06d2ff 343}
b40da9a7 344
ab06d2ff 345/* Returns true if the KEY is the same as that associated with
346 SLOT. */
347
348static int
349machopic_indirection_eq (const void *slot, const void *key)
350{
225ab426 351 return strcmp (((const machopic_indirection *) slot)->ptr_name,
352 (const char *) key) == 0;
ab06d2ff 353}
80d725d7 354
ab06d2ff 355/* Return the name of the non-lazy pointer (if STUB_P is false) or
356 stub (if STUB_B is true) corresponding to the given name. */
8dfaa51b 357
ab06d2ff 358const char *
359machopic_indirection_name (rtx sym_ref, bool stub_p)
360{
361 char *buffer;
362 const char *name = XSTR (sym_ref, 0);
6d3abfc0 363 size_t namelen = strlen (name);
ab06d2ff 364 machopic_indirection *p;
6d3abfc0 365 void ** slot;
524eafbc 366 bool saw_star = false;
367 bool needs_quotes;
368 const char *suffix;
369 const char *prefix = user_label_prefix;
370 const char *quote = "";
b359164b 371 tree id;
372
373 id = maybe_get_identifier (name);
374 if (id)
375 {
376 tree id_orig = id;
377
378 while (IDENTIFIER_TRANSPARENT_ALIAS (id))
379 id = TREE_CHAIN (id);
380 if (id != id_orig)
381 {
382 name = IDENTIFIER_POINTER (id);
383 namelen = strlen (name);
384 }
385 }
b215c058 386
524eafbc 387 if (name[0] == '*')
8dfaa51b 388 {
524eafbc 389 saw_star = true;
390 prefix = "";
391 ++name;
392 --namelen;
ab06d2ff 393 }
524eafbc 394
395 needs_quotes = name_needs_quotes (name);
396 if (needs_quotes)
ab06d2ff 397 {
524eafbc 398 quote = "\"";
8dfaa51b 399 }
400
524eafbc 401 if (stub_p)
402 suffix = STUB_SUFFIX;
403 else
404 suffix = NON_LAZY_POINTER_SUFFIX;
405
225ab426 406 buffer = XALLOCAVEC (char, strlen ("&L")
524eafbc 407 + strlen (prefix)
408 + namelen
409 + strlen (suffix)
410 + 2 * strlen (quote)
411 + 1 /* '\0' */);
412
413 /* Construct the name of the non-lazy pointer or stub. */
414 sprintf (buffer, "&%sL%s%s%s%s", quote, prefix, name, suffix, quote);
415
6d3abfc0 416 if (!machopic_indirections)
b215c058 417 machopic_indirections = htab_create_ggc (37,
6d3abfc0 418 machopic_indirection_hash,
419 machopic_indirection_eq,
420 /*htab_del=*/NULL);
b215c058 421
6d3abfc0 422 slot = htab_find_slot_with_hash (machopic_indirections, buffer,
423 htab_hash_string (buffer), INSERT);
424 if (*slot)
425 {
426 p = (machopic_indirection *) *slot;
427 }
428 else
ab06d2ff 429 {
ab06d2ff 430 p = (machopic_indirection *) ggc_alloc (sizeof (machopic_indirection));
431 p->symbol = sym_ref;
6d3abfc0 432 p->ptr_name = xstrdup (buffer);
ab06d2ff 433 p->stub_p = stub_p;
6d3abfc0 434 p->used = false;
435 *slot = p;
ab06d2ff 436 }
b215c058 437
6d3abfc0 438 return p->ptr_name;
80d725d7 439}
440
ab06d2ff 441/* Return the name of the stub for the mcount function. */
80d725d7 442
ab06d2ff 443const char*
444machopic_mcount_stub_name (void)
80d725d7 445{
620dd87a 446 rtx symbol = gen_rtx_SYMBOL_REF (Pmode, "*mcount");
447 return machopic_indirection_name (symbol, /*stub_p=*/true);
80d725d7 448}
449
ab06d2ff 450/* If NAME is the name of a stub or a non-lazy pointer , mark the stub
451 or non-lazy pointer as used -- and mark the object to which the
452 pointer/stub refers as used as well, since the pointer/stub will
453 emit a reference to it. */
454
80d725d7 455void
ab06d2ff 456machopic_validate_stub_or_non_lazy_ptr (const char *name)
80d725d7 457{
ab06d2ff 458 machopic_indirection *p;
b215c058 459
460 p = ((machopic_indirection *)
6d3abfc0 461 (htab_find_with_hash (machopic_indirections, name,
462 htab_hash_string (name))));
463 if (p && ! p->used)
ab06d2ff 464 {
6d3abfc0 465 const char *real_name;
466 tree id;
b215c058 467
6d3abfc0 468 p->used = true;
469
09e39ca0 470 /* Do what output_addr_const will do when we actually call it. */
471 if (SYMBOL_REF_DECL (p->symbol))
472 mark_decl_referenced (SYMBOL_REF_DECL (p->symbol));
473
474 real_name = targetm.strip_name_encoding (XSTR (p->symbol, 0));
b215c058 475
6d3abfc0 476 id = maybe_get_identifier (real_name);
477 if (id)
478 mark_referenced (id);
ab06d2ff 479 }
80d725d7 480}
481
482/* Transform ORIG, which may be any data source, to the corresponding
483 source using indirections. */
484
485rtx
b40da9a7 486machopic_indirect_data_reference (rtx orig, rtx reg)
80d725d7 487{
488 rtx ptr_ref = orig;
b40da9a7 489
80d725d7 490 if (! MACHOPIC_INDIRECT)
491 return orig;
492
493 if (GET_CODE (orig) == SYMBOL_REF)
494 {
ab06d2ff 495 int defined = machopic_data_defined_p (orig);
8f74ae7b 496
497 if (defined && MACHO_DYNAMIC_NO_PIC_P)
498 {
499#if defined (TARGET_TOC)
fed0a860 500 /* Create a new register for CSE opportunities. */
e1ba4a27 501 rtx hi_reg = (!can_create_pseudo_p () ? reg : gen_reg_rtx (Pmode));
fed0a860 502 emit_insn (gen_macho_high (hi_reg, orig));
503 emit_insn (gen_macho_low (reg, hi_reg, orig));
8f74ae7b 504#else
505 /* some other cpu -- writeme! */
27d2baef 506 gcc_unreachable ();
8f74ae7b 507#endif
508 return reg;
509 }
510 else if (defined)
80d725d7 511 {
5352873f 512#if defined (TARGET_TOC) || defined (HAVE_lo_sum)
a914e728 513 rtx offset = machopic_gen_offset (orig);
5352873f 514#endif
80d725d7 515
516#if defined (TARGET_TOC) /* i.e., PowerPC */
e1ba4a27 517 rtx hi_sum_reg = (!can_create_pseudo_p ()
518 ? reg
519 : gen_reg_rtx (Pmode));
80d725d7 520
27d2baef 521 gcc_assert (reg);
80d725d7 522
37ec5024 523 emit_insn (gen_rtx_SET (Pmode, hi_sum_reg,
524 gen_rtx_PLUS (Pmode, pic_offset_table_rtx,
525 gen_rtx_HIGH (Pmode, offset))));
526 emit_insn (gen_rtx_SET (Pmode, reg,
3b4a645f 527 gen_rtx_LO_SUM (Pmode, hi_sum_reg,
528 copy_rtx (offset))));
80d725d7 529
530 orig = reg;
531#else
532#if defined (HAVE_lo_sum)
27d2baef 533 gcc_assert (reg);
80d725d7 534
37ec5024 535 emit_insn (gen_rtx_SET (VOIDmode, reg,
536 gen_rtx_HIGH (Pmode, offset)));
537 emit_insn (gen_rtx_SET (VOIDmode, reg,
3b4a645f 538 gen_rtx_LO_SUM (Pmode, reg,
539 copy_rtx (offset))));
18b42941 540 emit_use (pic_offset_table_rtx);
80d725d7 541
37ec5024 542 orig = gen_rtx_PLUS (Pmode, pic_offset_table_rtx, reg);
80d725d7 543#endif
544#endif
545 return orig;
546 }
547
329f67ae 548 ptr_ref = (gen_rtx_SYMBOL_REF
b215c058 549 (Pmode,
ab06d2ff 550 machopic_indirection_name (orig, /*stub_p=*/false)));
80d725d7 551
7c266bb1 552 SYMBOL_REF_DATA (ptr_ref) = SYMBOL_REF_DATA (orig);
f9226418 553
e265a6da 554 ptr_ref = gen_const_mem (Pmode, ptr_ref);
329f67ae 555 machopic_define_symbol (ptr_ref);
80d725d7 556
557 return ptr_ref;
558 }
559 else if (GET_CODE (orig) == CONST)
560 {
561 rtx base, result;
562
563 /* legitimize both operands of the PLUS */
564 if (GET_CODE (XEXP (orig, 0)) == PLUS)
565 {
566 base = machopic_indirect_data_reference (XEXP (XEXP (orig, 0), 0),
567 reg);
568 orig = machopic_indirect_data_reference (XEXP (XEXP (orig, 0), 1),
569 (base == reg ? 0 : reg));
570 }
b40da9a7 571 else
80d725d7 572 return orig;
573
574 if (MACHOPIC_PURE && GET_CODE (orig) == CONST_INT)
b244d4c7 575 result = plus_constant (base, INTVAL (orig));
80d725d7 576 else
37ec5024 577 result = gen_rtx_PLUS (Pmode, base, orig);
80d725d7 578
80d725d7 579 if (MACHOPIC_JUST_INDIRECT && GET_CODE (base) == MEM)
580 {
581 if (reg)
582 {
583 emit_move_insn (reg, result);
584 result = reg;
585 }
586 else
587 {
588 result = force_reg (GET_MODE (result), result);
589 }
590 }
591
592 return result;
593
594 }
595 else if (GET_CODE (orig) == MEM)
596 XEXP (ptr_ref, 0) = machopic_indirect_data_reference (XEXP (orig, 0), reg);
597 /* When the target is i386, this code prevents crashes due to the
598 compiler's ignorance on how to move the PIC base register to
599 other registers. (The reload phase sometimes introduces such
600 insns.) */
601 else if (GET_CODE (orig) == PLUS
602 && GET_CODE (XEXP (orig, 0)) == REG
603 && REGNO (XEXP (orig, 0)) == PIC_OFFSET_TABLE_REGNUM
604#ifdef I386
605 /* Prevent the same register from being erroneously used
606 as both the base and index registers. */
607 && GET_CODE (XEXP (orig, 1)) == CONST
608#endif
609 && reg)
610 {
611 emit_move_insn (reg, XEXP (orig, 0));
612 XEXP (ptr_ref, 0) = reg;
613 }
614 return ptr_ref;
615}
616
80d725d7 617/* Transform TARGET (a MEM), which is a function call target, to the
618 corresponding symbol_stub if necessary. Return a new MEM. */
619
620rtx
b40da9a7 621machopic_indirect_call_target (rtx target)
80d725d7 622{
623 if (GET_CODE (target) != MEM)
624 return target;
625
b215c058 626 if (MACHOPIC_INDIRECT
ab06d2ff 627 && GET_CODE (XEXP (target, 0)) == SYMBOL_REF
628 && !(SYMBOL_REF_FLAGS (XEXP (target, 0))
629 & MACHO_SYMBOL_FLAG_DEFINED))
b40da9a7 630 {
ab06d2ff 631 rtx sym_ref = XEXP (target, 0);
b215c058 632 const char *stub_name = machopic_indirection_name (sym_ref,
ab06d2ff 633 /*stub_p=*/true);
634 enum machine_mode mode = GET_MODE (sym_ref);
b215c058 635
ab06d2ff 636 XEXP (target, 0) = gen_rtx_SYMBOL_REF (mode, stub_name);
7c266bb1 637 SYMBOL_REF_DATA (XEXP (target, 0)) = SYMBOL_REF_DATA (sym_ref);
b04fab2a 638 MEM_READONLY_P (target) = 1;
e265a6da 639 MEM_NOTRAP_P (target) = 1;
80d725d7 640 }
641
642 return target;
643}
644
645rtx
b40da9a7 646machopic_legitimize_pic_address (rtx orig, enum machine_mode mode, rtx reg)
80d725d7 647{
648 rtx pic_ref = orig;
649
8f74ae7b 650 if (! MACHOPIC_INDIRECT)
80d725d7 651 return orig;
652
653 /* First handle a simple SYMBOL_REF or LABEL_REF */
654 if (GET_CODE (orig) == LABEL_REF
655 || (GET_CODE (orig) == SYMBOL_REF
656 ))
657 {
658 /* addr(foo) = &func+(foo-func) */
80d725d7 659 orig = machopic_indirect_data_reference (orig, reg);
660
b40da9a7 661 if (GET_CODE (orig) == PLUS
80d725d7 662 && GET_CODE (XEXP (orig, 0)) == REG)
663 {
664 if (reg == 0)
665 return force_reg (mode, orig);
666
667 emit_move_insn (reg, orig);
668 return reg;
b40da9a7 669 }
80d725d7 670
80d725d7 671 if (GET_CODE (orig) == MEM)
672 {
673 if (reg == 0)
674 {
27d2baef 675 gcc_assert (!reload_in_progress);
676 reg = gen_reg_rtx (Pmode);
80d725d7 677 }
b40da9a7 678
80d725d7 679#ifdef HAVE_lo_sum
8f74ae7b 680 if (MACHO_DYNAMIC_NO_PIC_P
681 && (GET_CODE (XEXP (orig, 0)) == SYMBOL_REF
682 || GET_CODE (XEXP (orig, 0)) == LABEL_REF))
683 {
684#if defined (TARGET_TOC) /* ppc */
e1ba4a27 685 rtx temp_reg = (!can_create_pseudo_p ()
686 ? reg :
687 gen_reg_rtx (Pmode));
8f74ae7b 688 rtx asym = XEXP (orig, 0);
689 rtx mem;
690
301a5e52 691 emit_insn (gen_macho_high (temp_reg, asym));
e265a6da 692 mem = gen_const_mem (GET_MODE (orig),
3b4a645f 693 gen_rtx_LO_SUM (Pmode, temp_reg,
694 copy_rtx (asym)));
37ec5024 695 emit_insn (gen_rtx_SET (VOIDmode, reg, mem));
8f74ae7b 696#else
c4f8e01f 697 /* Some other CPU -- WriteMe! but right now there are no other
698 platforms that can use dynamic-no-pic */
27d2baef 699 gcc_unreachable ();
8f74ae7b 700#endif
701 pic_ref = reg;
702 }
703 else
b40da9a7 704 if (GET_CODE (XEXP (orig, 0)) == SYMBOL_REF
80d725d7 705 || GET_CODE (XEXP (orig, 0)) == LABEL_REF)
706 {
a914e728 707 rtx offset = machopic_gen_offset (XEXP (orig, 0));
80d725d7 708#if defined (TARGET_TOC) /* i.e., PowerPC */
709 /* Generating a new reg may expose opportunities for
710 common subexpression elimination. */
e1ba4a27 711 rtx hi_sum_reg = (!can_create_pseudo_p ()
712 ? reg
713 : gen_reg_rtx (Pmode));
37ec5024 714 rtx mem;
715 rtx insn;
716 rtx sum;
b215c058 717
37ec5024 718 sum = gen_rtx_HIGH (Pmode, offset);
719 if (! MACHO_DYNAMIC_NO_PIC_P)
720 sum = gen_rtx_PLUS (Pmode, pic_offset_table_rtx, sum);
80d725d7 721
37ec5024 722 emit_insn (gen_rtx_SET (Pmode, hi_sum_reg, sum));
723
e265a6da 724 mem = gen_const_mem (GET_MODE (orig),
b215c058 725 gen_rtx_LO_SUM (Pmode,
3b4a645f 726 hi_sum_reg,
727 copy_rtx (offset)));
37ec5024 728 insn = emit_insn (gen_rtx_SET (VOIDmode, reg, mem));
24153880 729 set_unique_reg_note (insn, REG_EQUAL, pic_ref);
37ec5024 730
731 pic_ref = reg;
80d725d7 732#else
18b42941 733 emit_use (gen_rtx_REG (Pmode, PIC_OFFSET_TABLE_REGNUM));
37ec5024 734
735 emit_insn (gen_rtx_SET (VOIDmode, reg,
736 gen_rtx_HIGH (Pmode,
b215c058 737 gen_rtx_CONST (Pmode,
37ec5024 738 offset))));
739 emit_insn (gen_rtx_SET (VOIDmode, reg,
740 gen_rtx_LO_SUM (Pmode, reg,
3b4a645f 741 gen_rtx_CONST (Pmode,
742 copy_rtx (offset)))));
37ec5024 743 pic_ref = gen_rtx_PLUS (Pmode,
744 pic_offset_table_rtx, reg);
80d725d7 745#endif
746 }
747 else
748#endif /* HAVE_lo_sum */
749 {
750 rtx pic = pic_offset_table_rtx;
751 if (GET_CODE (pic) != REG)
752 {
753 emit_move_insn (reg, pic);
754 pic = reg;
755 }
756#if 0
18b42941 757 emit_use (gen_rtx_REG (Pmode, PIC_OFFSET_TABLE_REGNUM));
80d725d7 758#endif
759
3e111a71 760 if (reload_in_progress)
3072d30e 761 df_set_regs_ever_live (REGNO (pic), true);
b06a3cc3 762 pic_ref = gen_rtx_PLUS (Pmode, pic,
a914e728 763 machopic_gen_offset (XEXP (orig, 0)));
80d725d7 764 }
b40da9a7 765
80d725d7 766#if !defined (TARGET_TOC)
80d725d7 767 emit_move_insn (reg, pic_ref);
e265a6da 768 pic_ref = gen_const_mem (GET_MODE (orig), reg);
7ba97778 769#endif
80d725d7 770 }
771 else
772 {
773
774#ifdef HAVE_lo_sum
b40da9a7 775 if (GET_CODE (orig) == SYMBOL_REF
80d725d7 776 || GET_CODE (orig) == LABEL_REF)
777 {
a914e728 778 rtx offset = machopic_gen_offset (orig);
80d725d7 779#if defined (TARGET_TOC) /* i.e., PowerPC */
780 rtx hi_sum_reg;
781
782 if (reg == 0)
783 {
27d2baef 784 gcc_assert (!reload_in_progress);
785 reg = gen_reg_rtx (Pmode);
80d725d7 786 }
b40da9a7 787
80d725d7 788 hi_sum_reg = reg;
789
37ec5024 790 emit_insn (gen_rtx_SET (Pmode, hi_sum_reg,
791 (MACHO_DYNAMIC_NO_PIC_P)
792 ? gen_rtx_HIGH (Pmode, offset)
793 : gen_rtx_PLUS (Pmode,
794 pic_offset_table_rtx,
b215c058 795 gen_rtx_HIGH (Pmode,
37ec5024 796 offset))));
797 emit_insn (gen_rtx_SET (VOIDmode, reg,
798 gen_rtx_LO_SUM (Pmode,
3b4a645f 799 hi_sum_reg,
800 copy_rtx (offset))));
80d725d7 801 pic_ref = reg;
802#else
37ec5024 803 emit_insn (gen_rtx_SET (VOIDmode, reg,
804 gen_rtx_HIGH (Pmode, offset)));
805 emit_insn (gen_rtx_SET (VOIDmode, reg,
3b4a645f 806 gen_rtx_LO_SUM (Pmode, reg,
807 copy_rtx (offset))));
37ec5024 808 pic_ref = gen_rtx_PLUS (Pmode,
809 pic_offset_table_rtx, reg);
80d725d7 810#endif
811 }
812 else
813#endif /* HAVE_lo_sum */
814 {
7a5893fc 815 if (REG_P (orig)
816 || GET_CODE (orig) == SUBREG)
80d725d7 817 {
818 return orig;
819 }
820 else
821 {
822 rtx pic = pic_offset_table_rtx;
823 if (GET_CODE (pic) != REG)
824 {
825 emit_move_insn (reg, pic);
826 pic = reg;
827 }
828#if 0
18b42941 829 emit_use (pic_offset_table_rtx);
80d725d7 830#endif
3e111a71 831 if (reload_in_progress)
3072d30e 832 df_set_regs_ever_live (REGNO (pic), true);
37ec5024 833 pic_ref = gen_rtx_PLUS (Pmode,
834 pic,
a914e728 835 machopic_gen_offset (orig));
80d725d7 836 }
837 }
838 }
839
80d725d7 840 if (GET_CODE (pic_ref) != REG)
841 {
842 if (reg != 0)
843 {
844 emit_move_insn (reg, pic_ref);
845 return reg;
846 }
847 else
848 {
849 return force_reg (mode, pic_ref);
850 }
851 }
852 else
853 {
854 return pic_ref;
855 }
856 }
857
858 else if (GET_CODE (orig) == SYMBOL_REF)
859 return orig;
860
861 else if (GET_CODE (orig) == PLUS
862 && (GET_CODE (XEXP (orig, 0)) == MEM
863 || GET_CODE (XEXP (orig, 0)) == SYMBOL_REF
864 || GET_CODE (XEXP (orig, 0)) == LABEL_REF)
865 && XEXP (orig, 0) != pic_offset_table_rtx
866 && GET_CODE (XEXP (orig, 1)) != REG)
b40da9a7 867
80d725d7 868 {
869 rtx base;
870 int is_complex = (GET_CODE (XEXP (orig, 0)) == MEM);
871
872 base = machopic_legitimize_pic_address (XEXP (orig, 0), Pmode, reg);
873 orig = machopic_legitimize_pic_address (XEXP (orig, 1),
874 Pmode, (base == reg ? 0 : reg));
875 if (GET_CODE (orig) == CONST_INT)
876 {
b244d4c7 877 pic_ref = plus_constant (base, INTVAL (orig));
80d725d7 878 is_complex = 1;
879 }
880 else
37ec5024 881 pic_ref = gen_rtx_PLUS (Pmode, base, orig);
80d725d7 882
80d725d7 883 if (reg && is_complex)
884 {
885 emit_move_insn (reg, pic_ref);
886 pic_ref = reg;
887 }
888 /* Likewise, should we set special REG_NOTEs here? */
889 }
890
891 else if (GET_CODE (orig) == CONST)
892 {
893 return machopic_legitimize_pic_address (XEXP (orig, 0), Pmode, reg);
894 }
895
896 else if (GET_CODE (orig) == MEM
897 && GET_CODE (XEXP (orig, 0)) == SYMBOL_REF)
898 {
899 rtx addr = machopic_legitimize_pic_address (XEXP (orig, 0), Pmode, reg);
b04fab2a 900 addr = replace_equiv_address (orig, addr);
80d725d7 901 emit_move_insn (reg, addr);
902 pic_ref = reg;
903 }
904
905 return pic_ref;
906}
907
ab06d2ff 908/* Output the stub or non-lazy pointer in *SLOT, if it has been used.
909 DATA is the FILE* for assembly output. Called from
910 htab_traverse. */
80d725d7 911
ab06d2ff 912static int
913machopic_output_indirection (void **slot, void *data)
80d725d7 914{
ab06d2ff 915 machopic_indirection *p = *((machopic_indirection **) slot);
916 FILE *asm_out_file = (FILE *) data;
917 rtx symbol;
918 const char *sym_name;
919 const char *ptr_name;
b215c058 920
ab06d2ff 921 if (!p->used)
922 return 1;
80d725d7 923
ab06d2ff 924 symbol = p->symbol;
925 sym_name = XSTR (symbol, 0);
6d3abfc0 926 ptr_name = p->ptr_name;
b215c058 927
ab06d2ff 928 if (p->stub_p)
80d725d7 929 {
80d725d7 930 char *sym;
931 char *stub;
b359164b 932 tree id;
933
934 id = maybe_get_identifier (sym_name);
935 if (id)
936 {
937 tree id_orig = id;
938
939 while (IDENTIFIER_TRANSPARENT_ALIAS (id))
940 id = TREE_CHAIN (id);
941 if (id != id_orig)
942 sym_name = IDENTIFIER_POINTER (id);
943 }
80d725d7 944
225ab426 945 sym = XALLOCAVEC (char, strlen (sym_name) + 2);
80d725d7 946 if (sym_name[0] == '*' || sym_name[0] == '&')
947 strcpy (sym, sym_name + 1);
948 else if (sym_name[0] == '-' || sym_name[0] == '+')
b40da9a7 949 strcpy (sym, sym_name);
80d725d7 950 else
e9764bbb 951 sprintf (sym, "%s%s", user_label_prefix, sym_name);
80d725d7 952
225ab426 953 stub = XALLOCAVEC (char, strlen (ptr_name) + 2);
ab06d2ff 954 if (ptr_name[0] == '*' || ptr_name[0] == '&')
955 strcpy (stub, ptr_name + 1);
80d725d7 956 else
ab06d2ff 957 sprintf (stub, "%s%s", user_label_prefix, ptr_name);
80d725d7 958
09e39ca0 959 machopic_output_stub (asm_out_file, sym, stub);
80d725d7 960 }
f0ed0e8a 961 else if (! indirect_data (symbol)
8d806159 962 && (machopic_symbol_defined_p (symbol)
963 || SYMBOL_REF_LOCAL_P (symbol)))
80d725d7 964 {
2f14b1f9 965 switch_to_section (data_section);
ab06d2ff 966 assemble_align (GET_MODE_ALIGNMENT (Pmode));
967 assemble_label (ptr_name);
968 assemble_integer (gen_rtx_SYMBOL_REF (Pmode, sym_name),
969 GET_MODE_SIZE (Pmode),
970 GET_MODE_ALIGNMENT (Pmode), 1);
80d725d7 971 }
ab06d2ff 972 else
973 {
f0ed0e8a 974 rtx init = const0_rtx;
975
92e4ba7d 976 switch_to_section (darwin_sections[machopic_nl_symbol_ptr_section]);
cd186228 977
978 /* Mach-O symbols are passed around in code through indirect
979 references and the original symbol_ref hasn't passed through
980 the generic handling and reference-catching in
981 output_operand, so we need to manually mark weak references
982 as such. */
983 if (SYMBOL_REF_WEAK (symbol))
984 {
985 tree decl = SYMBOL_REF_DECL (symbol);
986 gcc_assert (DECL_P (decl));
987
988 if (decl != NULL_TREE
989 && DECL_EXTERNAL (decl) && TREE_PUBLIC (decl)
990 /* Handle only actual external-only definitions, not
991 e.g. extern inline code or variables for which
992 storage has been allocated. */
993 && !TREE_STATIC (decl))
994 {
995 fputs ("\t.weak_reference ", asm_out_file);
996 assemble_name (asm_out_file, sym_name);
997 fputc ('\n', asm_out_file);
998 }
999 }
1000
ab06d2ff 1001 assemble_name (asm_out_file, ptr_name);
1002 fprintf (asm_out_file, ":\n");
b215c058 1003
ab06d2ff 1004 fprintf (asm_out_file, "\t.indirect_symbol ");
1005 assemble_name (asm_out_file, sym_name);
1006 fprintf (asm_out_file, "\n");
b215c058 1007
f0ed0e8a 1008 /* Variables that are marked with MACHO_SYMBOL_STATIC need to
1009 have their symbol name instead of 0 in the second entry of
1010 the non-lazy symbol pointer data structure when they are
1011 defined. This allows the runtime to rebind newer instances
1012 of the translation unit with the original instance of the
d188633a 1013 symbol. */
f0ed0e8a 1014
1015 if ((SYMBOL_REF_FLAGS (symbol) & MACHO_SYMBOL_STATIC)
1016 && machopic_symbol_defined_p (symbol))
1017 init = gen_rtx_SYMBOL_REF (Pmode, sym_name);
1018
1019 assemble_integer (init, GET_MODE_SIZE (Pmode),
ab06d2ff 1020 GET_MODE_ALIGNMENT (Pmode), 1);
1021 }
b215c058 1022
ab06d2ff 1023 return 1;
1024}
1025
1026void
1027machopic_finish (FILE *asm_out_file)
1028{
1029 if (machopic_indirections)
f0ed0e8a 1030 htab_traverse_noresize (machopic_indirections,
ab06d2ff 1031 machopic_output_indirection,
1032 asm_out_file);
80d725d7 1033}
1034
b40da9a7 1035int
1036machopic_operand_p (rtx op)
80d725d7 1037{
1038 if (MACHOPIC_JUST_INDIRECT)
a914e728 1039 return (GET_CODE (op) == SYMBOL_REF
1040 && machopic_symbol_defined_p (op));
1041 else
1042 return (GET_CODE (op) == CONST
1043 && GET_CODE (XEXP (op, 0)) == UNSPEC
1044 && XINT (XEXP (op, 0), 1) == UNSPEC_MACHOPIC_OFFSET);
80d725d7 1045}
8dfaa51b 1046
1047/* This function records whether a given name corresponds to a defined
1048 or undefined function or variable, for machopic_classify_ident to
1049 use later. */
1050
1051void
b40da9a7 1052darwin_encode_section_info (tree decl, rtx rtl, int first ATTRIBUTE_UNUSED)
8dfaa51b 1053{
5a121a91 1054 rtx sym_ref;
8dfaa51b 1055
9af9a32c 1056 /* Do the standard encoding things first. */
1057 default_encode_section_info (decl, rtl, first);
1058
ab06d2ff 1059 if (TREE_CODE (decl) != FUNCTION_DECL && TREE_CODE (decl) != VAR_DECL)
1060 return;
9af9a32c 1061
ab06d2ff 1062 sym_ref = XEXP (rtl, 0);
1063 if (TREE_CODE (decl) == VAR_DECL)
1064 SYMBOL_REF_FLAGS (sym_ref) |= MACHO_SYMBOL_FLAG_VARIABLE;
1065
1066 if (!DECL_EXTERNAL (decl)
1dc74225 1067 && (!TREE_PUBLIC (decl) || !DECL_WEAK (decl))
0a3ecdc1 1068 && ! lookup_attribute ("weakref", DECL_ATTRIBUTES (decl))
8dfaa51b 1069 && ((TREE_STATIC (decl)
1070 && (!DECL_COMMON (decl) || !TREE_PUBLIC (decl)))
2f4fae92 1071 || (!DECL_COMMON (decl) && DECL_INITIAL (decl)
fd419242 1072 && DECL_INITIAL (decl) != error_mark_node)))
ab06d2ff 1073 SYMBOL_REF_FLAGS (sym_ref) |= MACHO_SYMBOL_FLAG_DEFINED;
f0ed0e8a 1074
d188633a 1075 if (! TREE_PUBLIC (decl))
f0ed0e8a 1076 SYMBOL_REF_FLAGS (sym_ref) |= MACHO_SYMBOL_STATIC;
e479d0de 1077}
01d15dc5 1078
9423c9b7 1079void
1080darwin_mark_decl_preserved (const char *name)
1081{
1082 fprintf (asm_out_file, ".no_dead_strip ");
1083 assemble_name (asm_out_file, name);
1084 fputc ('\n', asm_out_file);
1085}
1086
50e6797e 1087static section *
1088darwin_text_section (int reloc, int weak)
52470889 1089{
50e6797e 1090 if (reloc)
1091 return (weak
1092 ? darwin_sections[text_unlikely_coal_section]
1093 : unlikely_text_section ());
7def0f5c 1094 else
50e6797e 1095 return (weak
1096 ? darwin_sections[text_coal_section]
1097 : text_section);
1098}
1099
1100static section *
1101darwin_rodata_section (int weak)
1102{
1103 return (weak
1104 ? darwin_sections[const_coal_section]
1105 : darwin_sections[const_section]);
1106}
b40da9a7 1107
50e6797e 1108static section *
1109darwin_mergeable_string_section (tree exp,
1110 unsigned HOST_WIDE_INT align)
1111{
1112 if (flag_merge_constants
1113 && TREE_CODE (exp) == STRING_CST
1114 && TREE_CODE (TREE_TYPE (exp)) == ARRAY_TYPE
1115 && align <= 256
a7df40d7 1116 && (int_size_in_bytes (TREE_TYPE (exp))
1117 == TREE_STRING_LENGTH (exp))
b40da9a7 1118 && ((size_t) TREE_STRING_LENGTH (exp)
a814bad5 1119 == strlen (TREE_STRING_POINTER (exp)) + 1))
92e4ba7d 1120 return darwin_sections[cstring_section];
50e6797e 1121
1122 return readonly_data_section;
1123}
1124
4be92808 1125#ifndef HAVE_GAS_LITERAL16
1126#define HAVE_GAS_LITERAL16 0
1127#endif
1128
50e6797e 1129static section *
1130darwin_mergeable_constant_section (tree exp,
1131 unsigned HOST_WIDE_INT align)
1132{
1133 enum machine_mode mode = DECL_MODE (exp);
1134 unsigned int modesize = GET_MODE_BITSIZE (mode);
1135
1136 if (flag_merge_constants
1137 && mode != VOIDmode
1138 && mode != BLKmode
1139 && modesize <= align
1140 && align >= 8
1141 && align <= 256
1142 && (align & (align -1)) == 0)
52470889 1143 {
853b103d 1144 tree size = TYPE_SIZE_UNIT (TREE_TYPE (exp));
52470889 1145
50e6797e 1146 if (TREE_CODE (size) == INTEGER_CST
1147 && TREE_INT_CST_LOW (size) == 4
1148 && TREE_INT_CST_HIGH (size) == 0)
1149 return darwin_sections[literal4_section];
1150 else if (TREE_CODE (size) == INTEGER_CST
1151 && TREE_INT_CST_LOW (size) == 8
3ebc7dec 1152 && TREE_INT_CST_HIGH (size) == 0)
50e6797e 1153 return darwin_sections[literal8_section];
4be92808 1154 else if (HAVE_GAS_LITERAL16
1155 && TARGET_64BIT
50e6797e 1156 && TREE_CODE (size) == INTEGER_CST
1157 && TREE_INT_CST_LOW (size) == 16
1158 && TREE_INT_CST_HIGH (size) == 0)
1159 return darwin_sections[literal16_section];
1160 else
1161 return readonly_data_section;
1162 }
1163
1164 return readonly_data_section;
1165}
1166
0b769e4a 1167int
1168machopic_reloc_rw_mask (void)
1169{
1170 return MACHOPIC_INDIRECT ? 3 : 0;
1171}
1172
50e6797e 1173section *
1174machopic_select_section (tree decl,
1175 int reloc,
1176 unsigned HOST_WIDE_INT align)
1177{
1178 bool weak = (DECL_P (decl)
1179 && DECL_WEAK (decl)
1180 && (lookup_attribute ("weak", DECL_ATTRIBUTES (decl))
1181 || ! lookup_attribute ("weak_import",
1182 DECL_ATTRIBUTES (decl))));
50e6797e 1183 section *base_section;
1184
c6134e23 1185 switch (categorize_decl_for_section (decl, reloc))
50e6797e 1186 {
1187 case SECCAT_TEXT:
1188 base_section = darwin_text_section (reloc, weak);
1189 break;
1190
1191 case SECCAT_RODATA:
1192 case SECCAT_SRODATA:
1193 base_section = darwin_rodata_section (weak);
1194 break;
1195
1196 case SECCAT_RODATA_MERGE_STR:
1197 base_section = darwin_mergeable_string_section (decl, align);
1198 break;
1199
1200 case SECCAT_RODATA_MERGE_STR_INIT:
1201 base_section = darwin_mergeable_string_section (DECL_INITIAL (decl), align);
1202 break;
1203
1204 case SECCAT_RODATA_MERGE_CONST:
1205 base_section = darwin_mergeable_constant_section (decl, align);
1206 break;
1207
1208 case SECCAT_DATA:
1209 case SECCAT_DATA_REL:
1210 case SECCAT_DATA_REL_LOCAL:
1211 case SECCAT_DATA_REL_RO:
1212 case SECCAT_DATA_REL_RO_LOCAL:
1213 case SECCAT_SDATA:
1214 case SECCAT_TDATA:
1215 case SECCAT_BSS:
1216 case SECCAT_SBSS:
1217 case SECCAT_TBSS:
1218 if (TREE_READONLY (decl) || TREE_CONSTANT (decl))
1219 base_section = weak ? darwin_sections[const_data_coal_section]
1220 : darwin_sections[const_data_section];
52470889 1221 else
50e6797e 1222 base_section = weak ? darwin_sections[data_coal_section] : data_section;
1223 break;
1224
1225 default:
1226 gcc_unreachable ();
52470889 1227 }
50e6797e 1228
1229 /* Darwin weird special cases. */
1230 if (TREE_CODE (decl) == CONSTRUCTOR
1231 && TREE_TYPE (decl)
1232 && TREE_CODE (TREE_TYPE (decl)) == RECORD_TYPE
1233 && TYPE_NAME (TREE_TYPE (decl)))
52470889 1234 {
50e6797e 1235 tree name = TYPE_NAME (TREE_TYPE (decl));
52470889 1236 if (TREE_CODE (name) == TYPE_DECL)
50e6797e 1237 name = DECL_NAME (name);
0616a948 1238
1239 if (!strcmp (IDENTIFIER_POINTER (name), "__builtin_ObjCString"))
50e6797e 1240 {
1241 if (flag_next_runtime)
1242 return darwin_sections[objc_constant_string_object_section];
1243 else
1244 return darwin_sections[objc_string_object_section];
1245 }
52470889 1246 else
50e6797e 1247 return base_section;
52470889 1248 }
50e6797e 1249 else if (TREE_CODE (decl) == VAR_DECL
1250 && DECL_NAME (decl)
1251 && TREE_CODE (DECL_NAME (decl)) == IDENTIFIER_NODE
1252 && IDENTIFIER_POINTER (DECL_NAME (decl))
1253 && !strncmp (IDENTIFIER_POINTER (DECL_NAME (decl)), "_OBJC_", 6))
52470889 1254 {
50e6797e 1255 const char *name = IDENTIFIER_POINTER (DECL_NAME (decl));
52470889 1256
1257 if (!strncmp (name, "_OBJC_CLASS_METHODS_", 20))
50e6797e 1258 return darwin_sections[objc_cls_meth_section];
52470889 1259 else if (!strncmp (name, "_OBJC_INSTANCE_METHODS_", 23))
50e6797e 1260 return darwin_sections[objc_inst_meth_section];
52470889 1261 else if (!strncmp (name, "_OBJC_CATEGORY_CLASS_METHODS_", 20))
50e6797e 1262 return darwin_sections[objc_cat_cls_meth_section];
52470889 1263 else if (!strncmp (name, "_OBJC_CATEGORY_INSTANCE_METHODS_", 23))
50e6797e 1264 return darwin_sections[objc_cat_inst_meth_section];
52470889 1265 else if (!strncmp (name, "_OBJC_CLASS_VARIABLES_", 22))
50e6797e 1266 return darwin_sections[objc_class_vars_section];
52470889 1267 else if (!strncmp (name, "_OBJC_INSTANCE_VARIABLES_", 25))
50e6797e 1268 return darwin_sections[objc_instance_vars_section];
52470889 1269 else if (!strncmp (name, "_OBJC_CLASS_PROTOCOLS_", 22))
50e6797e 1270 return darwin_sections[objc_cat_cls_meth_section];
52470889 1271 else if (!strncmp (name, "_OBJC_CLASS_NAME_", 17))
50e6797e 1272 return darwin_sections[objc_class_names_section];
52470889 1273 else if (!strncmp (name, "_OBJC_METH_VAR_NAME_", 20))
50e6797e 1274 return darwin_sections[objc_meth_var_names_section];
52470889 1275 else if (!strncmp (name, "_OBJC_METH_VAR_TYPE_", 20))
50e6797e 1276 return darwin_sections[objc_meth_var_types_section];
52470889 1277 else if (!strncmp (name, "_OBJC_CLASS_REFERENCES", 22))
50e6797e 1278 return darwin_sections[objc_cls_refs_section];
52470889 1279 else if (!strncmp (name, "_OBJC_CLASS_", 12))
50e6797e 1280 return darwin_sections[objc_class_section];
52470889 1281 else if (!strncmp (name, "_OBJC_METACLASS_", 16))
50e6797e 1282 return darwin_sections[objc_meta_class_section];
52470889 1283 else if (!strncmp (name, "_OBJC_CATEGORY_", 15))
50e6797e 1284 return darwin_sections[objc_category_section];
52470889 1285 else if (!strncmp (name, "_OBJC_SELECTOR_REFERENCES", 25))
50e6797e 1286 return darwin_sections[objc_selector_refs_section];
52470889 1287 else if (!strncmp (name, "_OBJC_SELECTOR_FIXUP", 20))
50e6797e 1288 return darwin_sections[objc_selector_fixup_section];
52470889 1289 else if (!strncmp (name, "_OBJC_SYMBOLS", 13))
50e6797e 1290 return darwin_sections[objc_symbols_section];
52470889 1291 else if (!strncmp (name, "_OBJC_MODULES", 13))
50e6797e 1292 return darwin_sections[objc_module_info_section];
c17b85ea 1293 else if (!strncmp (name, "_OBJC_IMAGE_INFO", 16))
50e6797e 1294 return darwin_sections[objc_image_info_section];
52470889 1295 else if (!strncmp (name, "_OBJC_PROTOCOL_INSTANCE_METHODS_", 32))
50e6797e 1296 return darwin_sections[objc_cat_inst_meth_section];
52470889 1297 else if (!strncmp (name, "_OBJC_PROTOCOL_CLASS_METHODS_", 29))
50e6797e 1298 return darwin_sections[objc_cat_cls_meth_section];
52470889 1299 else if (!strncmp (name, "_OBJC_PROTOCOL_REFS_", 20))
50e6797e 1300 return darwin_sections[objc_cat_cls_meth_section];
52470889 1301 else if (!strncmp (name, "_OBJC_PROTOCOL_", 15))
50e6797e 1302 return darwin_sections[objc_protocol_section];
52470889 1303 else
50e6797e 1304 return base_section;
52470889 1305 }
50e6797e 1306
1307 return base_section;
52470889 1308}
1309
bbfbe351 1310/* This can be called with address expressions as "rtx".
9cb8e99f 1311 They must go in "const". */
bbfbe351 1312
2f14b1f9 1313section *
b40da9a7 1314machopic_select_rtx_section (enum machine_mode mode, rtx x,
1315 unsigned HOST_WIDE_INT align ATTRIBUTE_UNUSED)
bbfbe351 1316{
f2e8796e 1317 if (GET_MODE_SIZE (mode) == 8
1318 && (GET_CODE (x) == CONST_INT
1319 || GET_CODE (x) == CONST_DOUBLE))
92e4ba7d 1320 return darwin_sections[literal8_section];
bbfbe351 1321 else if (GET_MODE_SIZE (mode) == 4
1322 && (GET_CODE (x) == CONST_INT
1323 || GET_CODE (x) == CONST_DOUBLE))
92e4ba7d 1324 return darwin_sections[literal4_section];
4be92808 1325 else if (HAVE_GAS_LITERAL16
1326 && TARGET_64BIT
7463cbdf 1327 && GET_MODE_SIZE (mode) == 16
3ebc7dec 1328 && (GET_CODE (x) == CONST_INT
2805e80b 1329 || GET_CODE (x) == CONST_DOUBLE
1330 || GET_CODE (x) == CONST_VECTOR))
3ebc7dec 1331 return darwin_sections[literal16_section];
459f8624 1332 else if (MACHOPIC_INDIRECT
76dece62 1333 && (GET_CODE (x) == SYMBOL_REF
1334 || GET_CODE (x) == CONST
1335 || GET_CODE (x) == LABEL_REF))
92e4ba7d 1336 return darwin_sections[const_data_section];
bbfbe351 1337 else
92e4ba7d 1338 return darwin_sections[const_section];
bbfbe351 1339}
1340
01d15dc5 1341void
b40da9a7 1342machopic_asm_out_constructor (rtx symbol, int priority ATTRIBUTE_UNUSED)
01d15dc5 1343{
8f74ae7b 1344 if (MACHOPIC_INDIRECT)
92e4ba7d 1345 switch_to_section (darwin_sections[mod_init_section]);
01d15dc5 1346 else
92e4ba7d 1347 switch_to_section (darwin_sections[constructor_section]);
09d688ff 1348 assemble_align (POINTER_SIZE);
1349 assemble_integer (symbol, POINTER_SIZE / BITS_PER_UNIT, POINTER_SIZE, 1);
01d15dc5 1350
8f74ae7b 1351 if (! MACHOPIC_INDIRECT)
01d15dc5 1352 fprintf (asm_out_file, ".reference .constructors_used\n");
1353}
1354
1355void
b40da9a7 1356machopic_asm_out_destructor (rtx symbol, int priority ATTRIBUTE_UNUSED)
01d15dc5 1357{
8f74ae7b 1358 if (MACHOPIC_INDIRECT)
92e4ba7d 1359 switch_to_section (darwin_sections[mod_term_section]);
01d15dc5 1360 else
92e4ba7d 1361 switch_to_section (darwin_sections[destructor_section]);
09d688ff 1362 assemble_align (POINTER_SIZE);
1363 assemble_integer (symbol, POINTER_SIZE / BITS_PER_UNIT, POINTER_SIZE, 1);
01d15dc5 1364
8f74ae7b 1365 if (! MACHOPIC_INDIRECT)
01d15dc5 1366 fprintf (asm_out_file, ".reference .destructors_used\n");
1367}
1f3233d1 1368
67c1e638 1369void
b40da9a7 1370darwin_globalize_label (FILE *stream, const char *name)
67c1e638 1371{
1372 if (!!strncmp (name, "_OBJC_", 6))
1373 default_globalize_label (stream, name);
1374}
1375
2f9fc8ef 1376void
b215c058 1377darwin_asm_named_section (const char *name,
537cd941 1378 unsigned int flags ATTRIBUTE_UNUSED,
1379 tree decl ATTRIBUTE_UNUSED)
2f9fc8ef 1380{
1dc74225 1381 fprintf (asm_out_file, "\t.section %s\n", name);
2f9fc8ef 1382}
1383
b215c058 1384void
1dc74225 1385darwin_unique_section (tree decl ATTRIBUTE_UNUSED, int reloc ATTRIBUTE_UNUSED)
2f9fc8ef 1386{
1dc74225 1387 /* Darwin does not use unique sections. */
2f9fc8ef 1388}
1389
5b182752 1390/* Handle __attribute__ ((apple_kext_compatibility)).
1391 This only applies to darwin kexts for 2.95 compatibility -- it shrinks the
1392 vtable for classes with this attribute (and their descendants) by not
1393 outputting the new 3.0 nondeleting destructor. This means that such
1394 objects CANNOT be allocated on the stack or as globals UNLESS they have
1395 a completely empty `operator delete'.
1396 Luckily, this fits in with the Darwin kext model.
1397
1398 This attribute also disables gcc3's potential overlaying of derived
1399 class data members on the padding at the end of the base class. */
1400
1401tree
1402darwin_handle_kext_attribute (tree *node, tree name,
1403 tree args ATTRIBUTE_UNUSED,
1404 int flags ATTRIBUTE_UNUSED,
1405 bool *no_add_attrs)
1406{
1407 /* APPLE KEXT stuff -- only applies with pure static C++ code. */
1408 if (! TARGET_KEXTABI)
1409 {
67a779df 1410 warning (0, "%qE 2.95 vtable-compatibility attribute applies "
1411 "only when compiling a kext", name);
5b182752 1412
1413 *no_add_attrs = true;
1414 }
1415 else if (TREE_CODE (*node) != RECORD_TYPE)
1416 {
67a779df 1417 warning (0, "%qE 2.95 vtable-compatibility attribute applies "
1418 "only to C++ classes", name);
5b182752 1419
1420 *no_add_attrs = true;
1421 }
1422
1423 return NULL_TREE;
1424}
1425
975d636c 1426/* Handle a "weak_import" attribute; arguments as in
1427 struct attribute_spec.handler. */
1428
1429tree
1430darwin_handle_weak_import_attribute (tree *node, tree name,
1431 tree ARG_UNUSED (args),
1432 int ARG_UNUSED (flags),
1433 bool * no_add_attrs)
1434{
d0ddd548 1435 if (TREE_CODE (*node) != FUNCTION_DECL && TREE_CODE (*node) != VAR_DECL)
975d636c 1436 {
67a779df 1437 warning (OPT_Wattributes, "%qE attribute ignored",
1438 name);
975d636c 1439 *no_add_attrs = true;
1440 }
1441 else
1442 declare_weak (*node);
1443
1444 return NULL_TREE;
1445}
a3952a36 1446
b215c058 1447/* Emit a label for an FDE, making it global and/or weak if appropriate.
ef1074f7 1448 The third parameter is nonzero if this is for exception handling.
1449 The fourth parameter is nonzero if this is just a placeholder for an
2f9fc8ef 1450 FDE that we are omitting. */
ef1074f7 1451
b215c058 1452void
ef1074f7 1453darwin_emit_unwind_label (FILE *file, tree decl, int for_eh, int empty)
2f9fc8ef 1454{
3469b0cf 1455 char *lab;
ef1074f7 1456
1457 if (! for_eh)
d08d29c0 1458 return;
2f9fc8ef 1459
e8f535c2 1460 lab = concat (IDENTIFIER_POINTER (DECL_ASSEMBLER_NAME (decl)), ".eh", NULL);
2f9fc8ef 1461
1462 if (TREE_PUBLIC (decl))
e8f535c2 1463 {
1464 targetm.asm_out.globalize_label (file, lab);
1465 if (DECL_VISIBILITY (decl) == VISIBILITY_HIDDEN)
1466 {
1467 fputs ("\t.private_extern ", file);
1468 assemble_name (file, lab);
1469 fputc ('\n', file);
1470 }
1471 }
2f9fc8ef 1472
1dc74225 1473 if (DECL_WEAK (decl))
e8f535c2 1474 {
1475 fputs ("\t.weak_definition ", file);
1476 assemble_name (file, lab);
1477 fputc ('\n', file);
1478 }
2f9fc8ef 1479
e8f535c2 1480 assemble_name (file, lab);
2f9fc8ef 1481 if (empty)
a3952a36 1482 {
e8f535c2 1483 fputs (" = 0\n", file);
a3952a36 1484
1485 /* Mark the absolute .eh and .eh1 style labels as needed to
1486 ensure that we don't dead code strip them and keep such
1487 labels from another instantiation point until we can fix this
1488 properly with group comdat support. */
e8f535c2 1489 darwin_mark_decl_preserved (lab);
a3952a36 1490 }
2f9fc8ef 1491 else
e8f535c2 1492 fputs (":\n", file);
2f9fc8ef 1493
1494 free (lab);
1495}
1496
b215c058 1497static GTY(()) unsigned long except_table_label_num;
1498
1499void
1500darwin_emit_except_table_label (FILE *file)
1501{
1502 char section_start_label[30];
1503
1504 ASM_GENERATE_INTERNAL_LABEL (section_start_label, "GCC_except_table",
1505 except_table_label_num++);
1506 ASM_OUTPUT_LABEL (file, section_start_label);
1507}
1508/* Generate a PC-relative reference to a Mach-O non-lazy-symbol. */
ef1074f7 1509
2f9fc8ef 1510void
1511darwin_non_lazy_pcrel (FILE *file, rtx addr)
1512{
2f9fc8ef 1513 const char *nlp_name;
1514
27d2baef 1515 gcc_assert (GET_CODE (addr) == SYMBOL_REF);
2f9fc8ef 1516
ab06d2ff 1517 nlp_name = machopic_indirection_name (addr, /*stub_p=*/false);
2f9fc8ef 1518 fputs ("\t.long\t", file);
1519 ASM_OUTPUT_LABELREF (file, nlp_name);
1520 fputs ("-.", file);
1521}
1522
2c5b6111 1523/* Emit an assembler directive to set visibility for a symbol. The
1524 only supported visibilities are VISIBILITY_DEFAULT and
1525 VISIBILITY_HIDDEN; the latter corresponds to Darwin's "private
1526 extern". There is no MACH-O equivalent of ELF's
1527 VISIBILITY_INTERNAL or VISIBILITY_PROTECTED. */
1528
b215c058 1529void
2c5b6111 1530darwin_assemble_visibility (tree decl, int vis)
1531{
1532 if (vis == VISIBILITY_DEFAULT)
1533 ;
1534 else if (vis == VISIBILITY_HIDDEN)
1535 {
1536 fputs ("\t.private_extern ", asm_out_file);
1537 assemble_name (asm_out_file,
1538 (IDENTIFIER_POINTER (DECL_ASSEMBLER_NAME (decl))));
1539 fputs ("\n", asm_out_file);
1540 }
1541 else
9b2d6d13 1542 warning (OPT_Wattributes, "internal and protected visibility attributes "
1543 "not supported in this configuration; ignored");
2c5b6111 1544}
1545
a08b74c8 1546/* Output a difference of two labels that will be an assembly time
1547 constant if the two labels are local. (.long lab1-lab2 will be
1548 very different if lab1 is at the boundary between two sections; it
1549 will be relocated according to the second section, not the first,
1550 so one ends up with a difference between labels in different
1551 sections, which is bad in the dwarf2 eh context for instance.) */
1552
1553static int darwin_dwarf_label_counter;
1554
1555void
85d3e3b5 1556darwin_asm_output_dwarf_delta (FILE *file, int size,
b40da9a7 1557 const char *lab1, const char *lab2)
a08b74c8 1558{
2f9fc8ef 1559 int islocaldiff = (lab1[0] == '*' && lab1[1] == 'L'
1560 && lab2[0] == '*' && lab2[1] == 'L');
dcf3c3f7 1561 const char *directive = (size == 8 ? ".quad" : ".long");
a08b74c8 1562
1563 if (islocaldiff)
1564 fprintf (file, "\t.set L$set$%d,", darwin_dwarf_label_counter);
1565 else
85d3e3b5 1566 fprintf (file, "\t%s\t", directive);
48126f0c 1567 assemble_name_raw (file, lab1);
a08b74c8 1568 fprintf (file, "-");
48126f0c 1569 assemble_name_raw (file, lab2);
a08b74c8 1570 if (islocaldiff)
85d3e3b5 1571 fprintf (file, "\n\t%s L$set$%d", directive, darwin_dwarf_label_counter++);
a08b74c8 1572}
d08d29c0 1573
1574/* Output labels for the start of the DWARF sections if necessary. */
1575void
1576darwin_file_start (void)
1577{
1578 if (write_symbols == DWARF2_DEBUG)
1579 {
f720b7c6 1580 static const char * const debugnames[] =
d08d29c0 1581 {
1582 DEBUG_FRAME_SECTION,
1583 DEBUG_INFO_SECTION,
1584 DEBUG_ABBREV_SECTION,
1585 DEBUG_ARANGES_SECTION,
1586 DEBUG_MACINFO_SECTION,
1587 DEBUG_LINE_SECTION,
1588 DEBUG_LOC_SECTION,
1589 DEBUG_PUBNAMES_SECTION,
af84796a 1590 DEBUG_PUBTYPES_SECTION,
d08d29c0 1591 DEBUG_STR_SECTION,
1592 DEBUG_RANGES_SECTION
1593 };
1594 size_t i;
1595
1596 for (i = 0; i < ARRAY_SIZE (debugnames); i++)
1597 {
1598 int namelen;
1599
1600 switch_to_section (get_section (debugnames[i], SECTION_DEBUG, NULL));
f720b7c6 1601
d08d29c0 1602 gcc_assert (strncmp (debugnames[i], "__DWARF,", 8) == 0);
1603 gcc_assert (strchr (debugnames[i] + 8, ','));
f720b7c6 1604
d08d29c0 1605 namelen = strchr (debugnames[i] + 8, ',') - (debugnames[i] + 8);
1606 fprintf (asm_out_file, "Lsection%.*s:\n", namelen, debugnames[i] + 8);
1607 }
1608 }
1609}
1610
1611/* Output an offset in a DWARF section on Darwin. On Darwin, DWARF section
1612 offsets are not represented using relocs in .o files; either the
1613 section never leaves the .o file, or the linker or other tool is
1614 responsible for parsing the DWARF and updating the offsets. */
1615
1616void
1617darwin_asm_output_dwarf_offset (FILE *file, int size, const char * lab,
1618 section *base)
1619{
1620 char sname[64];
1621 int namelen;
f720b7c6 1622
d08d29c0 1623 gcc_assert (base->common.flags & SECTION_NAMED);
1624 gcc_assert (strncmp (base->named.name, "__DWARF,", 8) == 0);
1625 gcc_assert (strchr (base->named.name + 8, ','));
1626
1627 namelen = strchr (base->named.name + 8, ',') - (base->named.name + 8);
1628 sprintf (sname, "*Lsection%.*s", namelen, base->named.name + 8);
1629 darwin_asm_output_dwarf_delta (file, size, lab, sname);
1630}
a08b74c8 1631
f6940372 1632void
b40da9a7 1633darwin_file_end (void)
f6940372 1634{
1635 machopic_finish (asm_out_file);
1636 if (strcmp (lang_hooks.name, "GNU C++") == 0)
1637 {
92e4ba7d 1638 switch_to_section (darwin_sections[constructor_section]);
1639 switch_to_section (darwin_sections[destructor_section]);
f6940372 1640 ASM_OUTPUT_ALIGN (asm_out_file, 1);
1641 }
9e96724e 1642 fprintf (asm_out_file, "\t.subsections_via_symbols\n");
f6940372 1643}
1644
5b182752 1645/* TODO: Add a language hook for identifying if a decl is a vtable. */
1646#define DARWIN_VTABLE_P(DECL) 0
1647
0765d519 1648/* Cross-module name binding. Darwin does not support overriding
5b182752 1649 functions at dynamic-link time, except for vtables in kexts. */
0765d519 1650
1651bool
a9f1838b 1652darwin_binds_local_p (const_tree decl)
0765d519 1653{
5b182752 1654 return default_binds_local_p_1 (decl,
1655 TARGET_KEXTABI && DARWIN_VTABLE_P (decl));
0765d519 1656}
1657
f13da317 1658#if 0
1659/* See TARGET_ASM_OUTPUT_ANCHOR for why we can't do this yet. */
8c27005f 1660/* The Darwin's implementation of TARGET_ASM_OUTPUT_ANCHOR. Define the
1661 anchor relative to ".", the current section position. We cannot use
1662 the default one because ASM_OUTPUT_DEF is wrong for Darwin. */
1663
1664void
1665darwin_asm_output_anchor (rtx symbol)
1666{
1667 fprintf (asm_out_file, "\t.set\t");
1668 assemble_name (asm_out_file, XSTR (symbol, 0));
1669 fprintf (asm_out_file, ", . + " HOST_WIDE_INT_PRINT_DEC "\n",
1670 SYMBOL_REF_BLOCK_OFFSET (symbol));
1671}
f13da317 1672#endif
8c27005f 1673
f720b7c6 1674/* Set the darwin specific attributes on TYPE. */
1675void
1676darwin_set_default_type_attributes (tree type)
1677{
1678 if (darwin_ms_struct
1679 && TREE_CODE (type) == RECORD_TYPE)
1680 TYPE_ATTRIBUTES (type) = tree_cons (get_identifier ("ms_struct"),
1681 NULL_TREE,
1682 TYPE_ATTRIBUTES (type));
1683}
1684
26ac986e 1685/* True, iff we're generating code for loadable kernel extensions. */
5b182752 1686
1687bool
1688darwin_kextabi_p (void) {
1689 return flag_apple_kext;
1690}
1691
1692void
1693darwin_override_options (void)
1694{
5b182752 1695 if (flag_mkernel || flag_apple_kext)
1696 {
1697 /* -mkernel implies -fapple-kext for C++ */
1698 if (strcmp (lang_hooks.name, "GNU C++") == 0)
1699 flag_apple_kext = 1;
1700
1701 flag_no_common = 1;
1702
1703 /* No EH in kexts. */
1704 flag_exceptions = 0;
1705 /* No -fnon-call-exceptions data in kexts. */
1706 flag_non_call_exceptions = 0;
1707 }
d53bb226 1708 if (flag_var_tracking
fe6e8926 1709 && strverscmp (darwin_macosx_version_min, "10.5") >= 0
ffdfb40f 1710 && debug_info_level >= DINFO_LEVEL_NORMAL
1711 && debug_hooks->var_location != do_nothing_debug_hooks.var_location)
d53bb226 1712 flag_var_tracking_uninit = 1;
5b182752 1713}
1714
6aee6ac8 1715/* Add $LDBL128 suffix to long double builtins. */
1716
1717static void
1718darwin_patch_builtin (int fncode)
1719{
1720 tree fn = built_in_decls[fncode];
1721 tree sym;
1722 char *newname;
1723
1724 if (!fn)
1725 return;
1726
1727 sym = DECL_ASSEMBLER_NAME (fn);
d243d9c9 1728 newname = ACONCAT (("_", IDENTIFIER_POINTER (sym), "$LDBL128", NULL));
1729
6aee6ac8 1730 set_user_assembler_name (fn, newname);
6aee6ac8 1731
1732 fn = implicit_built_in_decls[fncode];
1733 if (fn)
1734 set_user_assembler_name (fn, newname);
6aee6ac8 1735}
1736
1737void
1738darwin_patch_builtins (void)
1739{
1740 if (LONG_DOUBLE_TYPE_SIZE != 128)
1741 return;
1742
1743#define PATCH_BUILTIN(fncode) darwin_patch_builtin (fncode);
d243d9c9 1744#define PATCH_BUILTIN_NO64(fncode) \
1745 if (!TARGET_64BIT) \
6aee6ac8 1746 darwin_patch_builtin (fncode);
d243d9c9 1747#define PATCH_BUILTIN_VARIADIC(fncode) \
1748 if (!TARGET_64BIT \
6aee6ac8 1749 && (strverscmp (darwin_macosx_version_min, "10.3.9") >= 0)) \
1750 darwin_patch_builtin (fncode);
1751#include "darwin-ppc-ldouble-patch.def"
1752#undef PATCH_BUILTIN
1753#undef PATCH_BUILTIN_NO64
1754#undef PATCH_BUILTIN_VARIADIC
1755}
1756
1757
1f3233d1 1758#include "gt-darwin.h"