summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--cginstructions.c185
1 files changed, 167 insertions, 18 deletions
diff --git a/cginstructions.c b/cginstructions.c
index a3960ef..af6ae73 100644
--- a/cginstructions.c
+++ b/cginstructions.c
@@ -3650,13 +3650,23 @@ static void ccall_load_string_or_array_offset (int offset,int c_parameter_n,int
}
#endif
+#if defined (THREAD64) && (defined (LINUX_ELF) || defined (MACH_O64))
+LABEL *pthread_getspecific_label=NULL;
+#endif
+
#ifdef THREAD32
# define SAVED_A_STACK_P_OFFSET 12
+# define STRING_tlsp_tls_index "tlsp_tls_index"
#else
# ifdef THREAD64
# define SAVED_HEAP_P_OFFSET 24
# define SAVED_R15_OFFSET 32
# define SAVED_A_STACK_P_OFFSET 40
+# ifdef MACH_O64
+# define STRING_tlsp_tls_index "_tlsp_tls_index"
+# else
+# define STRING_tlsp_tls_index "tlsp_tls_index"
+# endif
# endif
#endif
@@ -4563,7 +4573,7 @@ void code_ccall (char *c_function_name,char *s,int length)
# ifdef THREAD64
if (save_state_in_global_variables){
if (tlsp_tls_index_label==NULL)
- tlsp_tls_index_label=enter_label ("tlsp_tls_index",IMPORT_LABEL);
+ tlsp_tls_index_label=enter_label (STRING_tlsp_tls_index,IMPORT_LABEL);
# ifndef LINUX_ELF
i_move_r_id (A_STACK_POINTER,SAVED_A_STACK_P_OFFSET,-4/*R9*/);
i_move_r_id (-7/*RDI*/,SAVED_HEAP_P_OFFSET,-4/*R9*/);
@@ -4971,6 +4981,38 @@ void code_ccall (char *c_function_name,char *s,int length)
i_jsr_l (label,0);
else
i_jsr_r (function_address_reg);
+
+# ifdef THREAD64
+ if (save_state_in_global_variables){
+ if (tlsp_tls_index_label==NULL)
+ tlsp_tls_index_label=enter_label (STRING_tlsp_tls_index,IMPORT_LABEL);
+ if (pthread_getspecific_label==NULL)
+ pthread_getspecific_label=enter_label ("_pthread_getspecific",IMPORT_LABEL);
+
+ i_sub_i_r (16,B_STACK_POINTER);
+
+ switch (result){
+ case 'V':
+ i_move_l_r (tlsp_tls_index_label,-7/*RDI*/);
+ i_jsr_l (pthread_getspecific_label,0);
+ i_move_r_r (REGISTER_D0,-4/*R9*/);
+ break;
+ case 'R':
+ i_fmove_fr_id (0,0,B_STACK_POINTER);
+ i_move_l_r (tlsp_tls_index_label,-7/*RDI*/);
+ i_jsr_l (pthread_getspecific_label,0);
+ i_move_r_r (REGISTER_D0,-4/*R9*/);
+ i_fmove_id_fr (0,B_STACK_POINTER,0);
+ break;
+ default:
+ i_move_r_id (REGISTER_D0,0,B_STACK_POINTER);
+ i_move_l_r (tlsp_tls_index_label,-7/*RDI*/);
+ i_jsr_l (pthread_getspecific_label,0);
+ i_move_r_r (REGISTER_D0,-4/*R9*/);
+ i_move_id_r (0,B_STACK_POINTER,REGISTER_D0);
+ }
+ }
+# endif
if (c_offset_before_pushing_arguments-(b_result_offset+a_result_offset)==0)
i_move_r_r (REGISTER_RBP,B_STACK_POINTER);
@@ -4981,6 +5023,17 @@ void code_ccall (char *c_function_name,char *s,int length)
i_move_r_r (a_stack_pointer,A_STACK_POINTER);
i_move_r_r (heap_pointer,HEAP_POINTER);
i_move_id_r (-16,REGISTER_RBP,-4/*R9*/);
+ } else {
+# ifdef THREAD64
+ i_move_id_r (SAVED_A_STACK_P_OFFSET,-4/*R9*/,A_STACK_POINTER);
+ i_move_id_r (SAVED_R15_OFFSET,-4/*R9*/,REGISTER_D7);
+ i_move_id_r (SAVED_HEAP_P_OFFSET,-4/*R9*/,-7/*RDI*/);
+# else
+ i_lea_l_i_r (saved_heap_p_label,0,-7/*RDI*/);
+ i_move_l_r (saved_a_stack_p_label,-6/*RSI*/);
+ i_move_id_r (8,-7/*RDI*/,REGISTER_D7);
+ i_move_id_r (0,-7/*RDI*/,-7/*RDI*/);
+# endif
}
# else /* for I486 && G_AI64 && ! (LINUX_ELF || MACHO_64) */
{
@@ -5225,24 +5278,23 @@ void code_ccall (char *c_function_name,char *s,int length)
# ifdef THREAD64
if (!save_state_in_global_variables)
i_move_id_r (-16,REGISTER_RBP,-4/*R9*/);
-# endif
-# endif
- /* for I486 && G_AI64 */
-
- if (save_state_in_global_variables){
-#ifdef THREAD64
+ else {
instruction_l_r (ILDTLSP,tlsp_tls_index_label,-4/*R9*/);
i_move_id_r (SAVED_A_STACK_P_OFFSET,-4/*R9*/,A_STACK_POINTER);
i_move_id_r (SAVED_R15_OFFSET,-4/*R9*/,REGISTER_D7);
i_move_id_r (SAVED_HEAP_P_OFFSET,-4/*R9*/,-7/*RDI*/);
-#else
+ }
+# else
+ if (save_state_in_global_variables){
i_lea_l_i_r (saved_heap_p_label,0,-7/*RDI*/);
i_move_l_r (saved_a_stack_p_label,-6/*RSI*/);
i_move_id_r (8,-7/*RDI*/,REGISTER_D7);
i_move_id_r (0,-7/*RDI*/,-7/*RDI*/);
-#endif
}
+# endif
+# endif
+ /* for I486 && G_AI64 */
if (callee_pops_arguments)
c_offset=c_offset_before_pushing_arguments;
@@ -5426,16 +5478,22 @@ static void save_registers_before_clean_call (void)
# ifdef G_AI64
i_sub_i_r (144,B_STACK_POINTER);
- i_move_r_id (-6/*RSI*/,136,B_STACK_POINTER);
# if defined (LINUX_ELF) || defined (MACH_O64)
+# ifndef THREAD64
i_move_r_r (-6/*RSI*/,3/*R11*/);
+# endif
+# else
+ i_move_r_id (-6/*RSI*/,136,B_STACK_POINTER);
# endif
# ifndef THREAD64
i_move_l_r (saved_a_stack_p_label,-6/*RSI*/);
# endif
- i_move_r_id (-7/*RDI*/,128,B_STACK_POINTER);
# if defined (LINUX_ELF) || defined (MACH_O64)
+# ifndef THREAD64
i_move_r_r (-7/*RDI*/,2/*R10*/);
+# endif
+# else
+ i_move_r_id (-7/*RDI*/,128,B_STACK_POINTER);
# endif
# ifndef THREAD64
i_lea_l_i_r (saved_heap_p_label,0,-7/*RDI*/);
@@ -5469,7 +5527,7 @@ static void save_registers_before_clean_call (void)
i_move_l_r (saved_heap_p_label,-5/*EDI*/);
# else
if (tlsp_tls_index_label==NULL)
- tlsp_tls_index_label=enter_label ("tlsp_tls_index",IMPORT_LABEL);
+ tlsp_tls_index_label=enter_label (STRING_tlsp_tls_index,IMPORT_LABEL);
i_move_r_id (-5/*EDI*/,12,B_STACK_POINTER);
instruction_l_r (ILDTLSP,tlsp_tls_index_label,-5/*EDI*/);
@@ -5514,7 +5572,7 @@ static void restore_registers_after_clean_call (void)
# ifdef THREAD64
if (tlsp_tls_index_label==NULL)
- tlsp_tls_index_label=enter_label ("tlsp_tls_index",IMPORT_LABEL);
+ tlsp_tls_index_label=enter_label (STRING_tlsp_tls_index,IMPORT_LABEL);
i_move_r_id (-6/*RSI*/,SAVED_A_STACK_P_OFFSET,-4/*R9*/);
i_move_r_id (-7/*RDI*/,SAVED_HEAP_P_OFFSET,-4/*R9*/);
@@ -5526,8 +5584,10 @@ static void restore_registers_after_clean_call (void)
i_move_r_id (REGISTER_D7,8,-6/*RSI*/);
# endif
+# if ! (defined (LINUX_ELF) || defined (MACH_O64))
i_move_id_r (136,B_STACK_POINTER,-6/*RSI*/);
i_move_id_r (128,B_STACK_POINTER,-7/*RDI*/);
+# endif
i_move_id_r ( 80,B_STACK_POINTER, 7/*R15*/);
i_move_id_r (120,B_STACK_POINTER, 1/*RBX*/);
@@ -5593,8 +5653,12 @@ static void restore_registers_after_clean_call (void)
static void insert_loads_of_r9_rsi_rdi_and_r15_before_call (struct basic_block *block_with_call)
{
/* hack to load r9, rsi, rdi and r15 before jmp or call */
- struct instruction *instruction1,*instruction2,*instruction3,*instruction4,*jmp_or_call_instruction,*old_second_last_instruction;
+# ifndef MACH_O64
+ struct instruction *instruction1;
+# endif
+ struct instruction *instruction2,*instruction3,*instruction4,*jmp_or_call_instruction,*old_second_last_instruction;
+# if !(defined (LINUX_ELF) || defined (MACH_O64))
/* LDTLSP tlsp_tls_index_label,r9 */
instruction1=(struct instruction*)fast_memory_allocate (sizeof (struct instruction)+2*sizeof (struct parameter));
@@ -5613,6 +5677,7 @@ static void insert_loads_of_r9_rsi_rdi_and_r15_before_call (struct basic_block *
else
old_second_last_instruction->instruction_next=instruction1;
instruction1->instruction_prev=old_second_last_instruction;
+# endif
/* i_move_id_r (SAVED_A_STACK_P_OFFSET,R9,-6 */ /*RSI*/ /*); */
@@ -5625,8 +5690,19 @@ static void insert_loads_of_r9_rsi_rdi_and_r15_before_call (struct basic_block *
instruction2->instruction_parameters[1].parameter_type=P_REGISTER;
instruction2->instruction_parameters[1].parameter_data.i=-6/*RSI*/;
+# if !(defined (LINUX_ELF) || defined (MACH_O64))
instruction1->instruction_next=instruction2;
instruction2->instruction_prev=instruction1;
+# else
+ jmp_or_call_instruction=block_with_call->block_last_instruction;
+ old_second_last_instruction=jmp_or_call_instruction->instruction_prev;
+
+ if (old_second_last_instruction==NULL)
+ block_with_call->block_instructions=instruction2;
+ else
+ old_second_last_instruction->instruction_next=instruction2;
+ instruction2->instruction_prev=old_second_last_instruction;
+# endif
/* i_move_id_r (SAVED_R15_OFFSET,R9,REGISTER_D7); */
@@ -5661,6 +5737,68 @@ static void insert_loads_of_r9_rsi_rdi_and_r15_before_call (struct basic_block *
}
#endif
+#if defined (THREAD64) && (defined (LINUX_ELF) || defined (MACH_O64))
+static void call_pthread_getspecific (int n_integer_parameters,int n_float_parameters)
+{
+ int float_parameter_n;
+
+ if (tlsp_tls_index_label==NULL)
+ tlsp_tls_index_label=enter_label (STRING_tlsp_tls_index,IMPORT_LABEL);
+ if (pthread_getspecific_label==NULL)
+ pthread_getspecific_label=enter_label ("_pthread_getspecific",IMPORT_LABEL);
+
+ if (n_float_parameters>8)
+ n_float_parameters=8;
+
+ switch (n_integer_parameters){
+ default: /* >=6 */
+ i_move_r_r (-4/*R9 */, 4/*R12*/);
+ case 5:
+ i_move_r_r (-3/*R8 */, 7/*R15*/);
+ case 4:
+ i_move_r_r (-1/*RCX*/, 6/*R14*/);
+ case 3:
+ i_move_r_r (-2/*RDX*/, 5/*R13*/);
+ case 2:
+ i_move_r_r (-6/*RSI*/,-5/*RBP*/);
+ case 1:
+ i_move_r_r (-7/*RDI*/, 1/*RBX*/);
+ case 0:
+ break;
+ }
+
+ i_sub_i_r (8+(n_float_parameters<<3),B_STACK_POINTER);
+
+ for (float_parameter_n=0; float_parameter_n<n_float_parameters; ++float_parameter_n)
+ i_fmove_fr_id (float_parameter_n,float_parameter_n<<3,B_STACK_POINTER);
+
+ i_move_l_r (tlsp_tls_index_label,-7/*RDI*/);
+ i_jsr_l (pthread_getspecific_label,0);
+
+ for (float_parameter_n=0; float_parameter_n<n_float_parameters; ++float_parameter_n)
+ i_fmove_id_fr (float_parameter_n<<3,B_STACK_POINTER,float_parameter_n);
+
+ i_move_r_r (REGISTER_D0,-4/*R9*/);
+
+ i_add_i_r (8+(n_float_parameters<<3),B_STACK_POINTER);
+
+ switch (n_integer_parameters){
+ default: /* >=6 */
+ case 5:
+ i_move_r_r ( 7/*R15*/,-3/*R8 */);
+ case 4:
+ i_move_r_r ( 6/*R14*/,-1/*RCX*/);
+ case 3:
+ i_move_r_r ( 5/*R13*/,-2/*RDX*/);
+ case 2:
+ i_move_r_r (-5/*RBP*/, 3/*R11*/);
+ case 1:
+ i_move_r_r ( 1/*RBX*/, 2/*R10*/);
+ case 0:
+ break;
+ }
+}
+#endif
#ifdef I486
# ifdef G_AI64
@@ -5821,7 +5959,7 @@ void code_centry (char *c_function_name,char *clean_function_label,char *s,int l
n_integer_and_float_parameters=n_integer_parameters+n_float_parameters;
#endif
-# if (defined (sparc) && !defined (SOLARIS)) || (defined (I486) && !defined (G_AI64) && !defined (LINUX_ELF)) || (defined (G_POWER) && !defined (LINUX_ELF)) || defined (MACH_O)
+# if (defined (sparc) && !defined (SOLARIS)) || (defined (I486) && !defined (LINUX_ELF) && !defined (G_AI64)) || (defined (G_POWER) && !defined (LINUX_ELF)) || defined (MACH_O) || defined (MACH_O64)
{
char label_name [202];
@@ -5871,6 +6009,10 @@ void code_centry (char *c_function_name,char *clean_function_label,char *s,int l
save_registers_before_clean_call();
+#if defined (THREAD64) && (defined (LINUX_ELF) || defined (MACH_O64))
+ call_pthread_getspecific (n_integer_parameters+n_string_or_array_parameters,n_float_parameters);
+#endif
+
#if defined (G_AI64)
if (n_string_or_array_parameters!=0){
int register_n;
@@ -5910,8 +6052,9 @@ void code_centry (char *c_function_name,char *clean_function_label,char *s,int l
int i,offset;
# ifdef THREAD64
+# if !(defined (LINUX_ELF) || defined (MACH_O64))
instruction_l_r (ILDTLSP,tlsp_tls_index_label,-4/*R9*/);
-
+# endif
i_move_id_r (SAVED_A_STACK_P_OFFSET,-4/*R9*/,A_STACK_POINTER);
i_move_id_r (SAVED_R15_OFFSET,-4/*R9*/,REGISTER_D7);
i_move_id_r (SAVED_HEAP_P_OFFSET,-4/*R9*/,-7/*RDI*/);
@@ -6020,7 +6163,7 @@ void code_centry (char *c_function_name,char *clean_function_label,char *s,int l
register_n = (n_integer_and_float_parameters-1)-n;
if (s[first_parameter_index+register_n]!='R')
-# ifdef LINUX_ELF
+# if defined (LINUX_ELF) || defined (MACH_O64)
{
switch (register_n){
case 0: register_n= 2/*R10 was RDI*/; break;
@@ -6028,7 +6171,13 @@ void code_centry (char *c_function_name,char *clean_function_label,char *s,int l
case 2: register_n=-2/*RDX*/; break;
case 3: register_n=-1/*RCX*/; break;
case 4: register_n=-3/*R8*/; break;
- case 5: register_n=-4/*R9*/; break;
+ case 5:
+# ifndef THREAD64
+ register_n=-4/*R9*/;
+# else
+ register_n= 4/*R12*/;
+# endif
+ break;
default: error ("error in centry");
}
s_push_b (g_g_register (register_n));