mirror of
				https://git.proxmox.com/git/qemu
				synced 2025-10-24 21:41:23 +00:00 
			
		
		
		
	added tcg_temp_free() and improved the handling of constants
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4544 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
		
							parent
							
								
									d7e4036e31
								
							
						
					
					
						commit
						e8996ee012
					
				
							
								
								
									
										164
									
								
								tcg/tcg-op.h
									
									
									
									
									
								
							
							
						
						
									
										164
									
								
								tcg/tcg-op.h
									
									
									
									
									
								
							| @ -178,88 +178,115 @@ static inline void tcg_gen_movi_i32(TCGv ret, int32_t arg) | ||||
| 
 | ||||
| static inline void tcg_gen_helper_0_0(void *func) | ||||
| { | ||||
|     TCGv t0; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx,  | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  0, NULL, 0, NULL); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_0_1(void *func, TCGv arg) | ||||
| { | ||||
|     TCGv t0; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  0, NULL, 1, &arg); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_0_2(void *func, TCGv arg1, TCGv arg2) | ||||
| { | ||||
|     TCGv args[2]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx,  | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  0, NULL, 2, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_0_3(void *func, | ||||
|                                       TCGv arg1, TCGv arg2, TCGv arg3) | ||||
| { | ||||
|     TCGv args[3]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     args[2] = arg3; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  0, NULL, 3, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_0_4(void *func, TCGv arg1, TCGv arg2, | ||||
|                                       TCGv arg3, TCGv arg4) | ||||
| { | ||||
|     TCGv args[4]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     args[2] = arg3; | ||||
|     args[3] = arg4; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  0, NULL, 4, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_1_0(void *func, TCGv ret) | ||||
| { | ||||
|     TCGv t0; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  1, &ret, 0, NULL); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_1_1(void *func, TCGv ret, TCGv arg1) | ||||
| { | ||||
|     TCGv t0; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  1, &ret, 1, &arg1); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_1_2(void *func, TCGv ret,  | ||||
|                                       TCGv arg1, TCGv arg2) | ||||
| { | ||||
|     TCGv args[2]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx,  | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  1, &ret, 2, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_1_3(void *func, TCGv ret, | ||||
|                                       TCGv arg1, TCGv arg2, TCGv arg3) | ||||
| { | ||||
|     TCGv args[3]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     args[2] = arg3; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  1, &ret, 3, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_helper_1_4(void *func, TCGv ret, | ||||
| @ -267,13 +294,16 @@ static inline void tcg_gen_helper_1_4(void *func, TCGv ret, | ||||
|                                       TCGv arg4) | ||||
| { | ||||
|     TCGv args[4]; | ||||
|     TCGv t0; | ||||
|     args[0] = arg1; | ||||
|     args[1] = arg2; | ||||
|     args[2] = arg3; | ||||
|     args[3] = arg4; | ||||
|     t0 = tcg_const_ptr((tcg_target_long)func); | ||||
|     tcg_gen_call(&tcg_ctx, | ||||
|                  tcg_const_ptr((tcg_target_long)func), TCG_HELPER_CALL_FLAGS, | ||||
|                  t0, TCG_HELPER_CALL_FLAGS, | ||||
|                  1, &ret, 4, args); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| /* 32 bit ops */ | ||||
| @ -329,7 +359,9 @@ static inline void tcg_gen_addi_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_add_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_add_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -344,7 +376,9 @@ static inline void tcg_gen_subi_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_sub_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_sub_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -361,7 +395,9 @@ static inline void tcg_gen_andi_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     } else if (arg2 == 0xffffffff) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_and_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_and_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -378,7 +414,9 @@ static inline void tcg_gen_ori_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     } else if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_or_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_or_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -393,7 +431,9 @@ static inline void tcg_gen_xori_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_xor_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_xor_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -407,7 +447,9 @@ static inline void tcg_gen_shli_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_shl_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_shl_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -421,7 +463,9 @@ static inline void tcg_gen_shri_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_shr_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_shr_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -435,7 +479,9 @@ static inline void tcg_gen_sari_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i32(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_sar_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|         TCGv t0 = tcg_const_i32(arg2); | ||||
|         tcg_gen_sar_i32(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -452,7 +498,9 @@ static inline void tcg_gen_mul_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_muli_i32(TCGv ret, TCGv arg1, int32_t arg2) | ||||
| { | ||||
|     tcg_gen_mul_i32(ret, arg1, tcg_const_i32(arg2)); | ||||
|     TCGv t0 = tcg_const_i32(arg2); | ||||
|     tcg_gen_mul_i32(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| #ifdef TCG_TARGET_HAS_div_i32 | ||||
| @ -482,6 +530,7 @@ static inline void tcg_gen_div_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I32); | ||||
|     tcg_gen_sari_i32(t0, arg1, 31); | ||||
|     tcg_gen_op5(INDEX_op_div2_i32, ret, t0, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_rem_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -490,6 +539,7 @@ static inline void tcg_gen_rem_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I32); | ||||
|     tcg_gen_sari_i32(t0, arg1, 31); | ||||
|     tcg_gen_op5(INDEX_op_div2_i32, t0, ret, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_divu_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -498,6 +548,7 @@ static inline void tcg_gen_divu_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I32); | ||||
|     tcg_gen_movi_i32(t0, 0); | ||||
|     tcg_gen_op5(INDEX_op_divu2_i32, ret, t0, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_remu_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -506,6 +557,7 @@ static inline void tcg_gen_remu_i32(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I32); | ||||
|     tcg_gen_movi_i32(t0, 0); | ||||
|     tcg_gen_op5(INDEX_op_divu2_i32, t0, ret, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| @ -608,7 +660,9 @@ static inline void tcg_gen_add_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_addi_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_add_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_add_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_sub_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -619,7 +673,9 @@ static inline void tcg_gen_sub_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_subi_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_sub_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_sub_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_and_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -713,11 +769,15 @@ static inline void tcg_gen_mul_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     tcg_gen_add_i32(TCGV_HIGH(t0), TCGV_HIGH(t0), t1); | ||||
|      | ||||
|     tcg_gen_mov_i64(ret, t0); | ||||
|     tcg_temp_free(t0); | ||||
|     tcg_temp_free(t1); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_muli_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_mul_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_mul_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_div_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -824,7 +884,9 @@ static inline void tcg_gen_add_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_addi_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_add_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_add_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_sub_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -834,7 +896,9 @@ static inline void tcg_gen_sub_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_subi_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_sub_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_sub_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_and_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -844,7 +908,9 @@ static inline void tcg_gen_and_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_andi_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_and_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_and_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_or_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -854,7 +920,9 @@ static inline void tcg_gen_or_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_ori_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_or_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_or_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_xor_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -864,7 +932,9 @@ static inline void tcg_gen_xor_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_xori_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_xor_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_xor_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_shl_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -877,7 +947,9 @@ static inline void tcg_gen_shli_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i64(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_shl_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|         TCGv t0 = tcg_const_i64(arg2); | ||||
|         tcg_gen_shl_i64(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -891,7 +963,9 @@ static inline void tcg_gen_shri_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i64(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_shr_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|         TCGv t0 = tcg_const_i64(arg2); | ||||
|         tcg_gen_shr_i64(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -905,7 +979,9 @@ static inline void tcg_gen_sari_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
|     if (arg2 == 0) { | ||||
|         tcg_gen_mov_i64(ret, arg1); | ||||
|     } else { | ||||
|         tcg_gen_sar_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|         TCGv t0 = tcg_const_i64(arg2); | ||||
|         tcg_gen_sar_i64(ret, arg1, t0); | ||||
|         tcg_temp_free(t0); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -922,7 +998,9 @@ static inline void tcg_gen_mul_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| 
 | ||||
| static inline void tcg_gen_muli_i64(TCGv ret, TCGv arg1, int64_t arg2) | ||||
| { | ||||
|     tcg_gen_mul_i64(ret, arg1, tcg_const_i64(arg2)); | ||||
|     TCGv t0 = tcg_const_i64(arg2); | ||||
|     tcg_gen_mul_i64(ret, arg1, t0); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| #ifdef TCG_TARGET_HAS_div_i64 | ||||
| @ -952,6 +1030,7 @@ static inline void tcg_gen_div_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I64); | ||||
|     tcg_gen_sari_i64(t0, arg1, 63); | ||||
|     tcg_gen_op5(INDEX_op_div2_i64, ret, t0, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_rem_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -960,6 +1039,7 @@ static inline void tcg_gen_rem_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I64); | ||||
|     tcg_gen_sari_i64(t0, arg1, 63); | ||||
|     tcg_gen_op5(INDEX_op_div2_i64, t0, ret, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_divu_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -968,6 +1048,7 @@ static inline void tcg_gen_divu_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I64); | ||||
|     tcg_gen_movi_i64(t0, 0); | ||||
|     tcg_gen_op5(INDEX_op_divu2_i64, ret, t0, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_remu_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
| @ -976,6 +1057,7 @@ static inline void tcg_gen_remu_i64(TCGv ret, TCGv arg1, TCGv arg2) | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I64); | ||||
|     tcg_gen_movi_i64(t0, 0); | ||||
|     tcg_gen_op5(INDEX_op_divu2_i64, t0, ret, arg1, t0, arg2); | ||||
|     tcg_temp_free(t0); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| @ -1030,6 +1112,8 @@ static inline void tcg_gen_bswap16_i32(TCGv ret, TCGv arg) | ||||
|     tcg_gen_andi_i32(t1, arg, 0x000000ff); | ||||
|     tcg_gen_shli_i32(t1, t1, 8); | ||||
|     tcg_gen_or_i32(ret, t0, t1); | ||||
|     tcg_temp_free(t0); | ||||
|     tcg_temp_free(t1); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| @ -1054,6 +1138,8 @@ static inline void tcg_gen_bswap_i32(TCGv ret, TCGv arg) | ||||
|      | ||||
|     tcg_gen_shri_i32(t1, arg, 24); | ||||
|     tcg_gen_or_i32(ret, t0, t1); | ||||
|     tcg_temp_free(t0); | ||||
|     tcg_temp_free(t1); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| @ -1121,6 +1207,8 @@ static inline void tcg_gen_bswap_i64(TCGv ret, TCGv arg) | ||||
|     tcg_gen_bswap_i32(t1, TCGV_HIGH(arg)); | ||||
|     tcg_gen_mov_i32(ret, t1); | ||||
|     tcg_gen_mov_i32(TCGV_HIGH(ret), t0); | ||||
|     tcg_temp_free(t0); | ||||
|     tcg_temp_free(t1); | ||||
| } | ||||
| #else | ||||
| 
 | ||||
| @ -1227,6 +1315,8 @@ static inline void tcg_gen_bswap_i64(TCGv ret, TCGv arg) | ||||
| 
 | ||||
|     tcg_gen_shri_i64(t1, arg, 56); | ||||
|     tcg_gen_or_i64(ret, t0, t1); | ||||
|     tcg_temp_free(t0); | ||||
|     tcg_temp_free(t1); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| @ -1237,7 +1327,9 @@ static inline void tcg_gen_neg_i32(TCGv ret, TCGv arg) | ||||
| #ifdef TCG_TARGET_HAS_neg_i32 | ||||
|     tcg_gen_op2(INDEX_op_neg_i32, ret, arg); | ||||
| #else | ||||
|     tcg_gen_sub_i32(ret, tcg_const_i32(0), arg); | ||||
|     TCGv t0 = tcg_const_i32(0); | ||||
|     tcg_gen_sub_i32(ret, t0, arg); | ||||
|     tcg_temp_free(t0); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| @ -1246,18 +1338,20 @@ static inline void tcg_gen_neg_i64(TCGv ret, TCGv arg) | ||||
| #ifdef TCG_TARGET_HAS_neg_i64 | ||||
|     tcg_gen_op2(INDEX_op_neg_i64, ret, arg); | ||||
| #else | ||||
|     tcg_gen_sub_i64(ret, tcg_const_i64(0), arg); | ||||
|     TCGv t0 = tcg_const_i64(0); | ||||
|     tcg_gen_sub_i64(ret, t0, arg); | ||||
|     tcg_temp_free(t0); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_not_i32(TCGv ret, TCGv arg) | ||||
| { | ||||
|     tcg_gen_xor_i32(ret, arg, tcg_const_i32(-1)); | ||||
|     tcg_gen_xori_i32(ret, arg, -1); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_not_i64(TCGv ret, TCGv arg) | ||||
| { | ||||
|     tcg_gen_xor_i64(ret, arg, tcg_const_i64(-1)); | ||||
|     tcg_gen_xori_i64(ret, arg, -1); | ||||
| } | ||||
| 
 | ||||
| static inline void tcg_gen_discard_i32(TCGv arg) | ||||
|  | ||||
							
								
								
									
										460
									
								
								tcg/tcg.c
									
									
									
									
									
								
							
							
						
						
									
										460
									
								
								tcg/tcg.c
									
									
									
									
									
								
							| @ -266,8 +266,11 @@ void tcg_set_macro_func(TCGContext *s, TCGMacroFunc *func) | ||||
| 
 | ||||
| void tcg_func_start(TCGContext *s) | ||||
| { | ||||
|     int i; | ||||
|     tcg_pool_reset(s); | ||||
|     s->nb_temps = s->nb_globals; | ||||
|     for(i = 0; i < TCG_TYPE_COUNT; i++) | ||||
|         s->first_free_temp[i] = -1; | ||||
|     s->labels = tcg_malloc(sizeof(TCGLabel) * TCG_MAX_LABELS); | ||||
|     s->nb_labels = 0; | ||||
|     s->current_frame_offset = s->frame_start; | ||||
| @ -301,7 +304,6 @@ TCGv tcg_global_reg_new(TCGType type, int reg, const char *name) | ||||
|     ts->type = type; | ||||
|     ts->fixed_reg = 1; | ||||
|     ts->reg = reg; | ||||
|     ts->val_type = TEMP_VAL_REG; | ||||
|     ts->name = name; | ||||
|     s->nb_globals++; | ||||
|     tcg_regset_set_reg(s->reserved_regs, reg); | ||||
| @ -327,7 +329,6 @@ TCGv tcg_global_reg2_new_hack(TCGType type, int reg1, int reg2, | ||||
|     ts->type = TCG_TYPE_I32; | ||||
|     ts->fixed_reg = 1; | ||||
|     ts->reg = reg1; | ||||
|     ts->val_type = TEMP_VAL_REG; | ||||
|     pstrcpy(buf, sizeof(buf), name); | ||||
|     pstrcat(buf, sizeof(buf), "_0"); | ||||
|     ts->name = strdup(buf); | ||||
| @ -337,7 +338,6 @@ TCGv tcg_global_reg2_new_hack(TCGType type, int reg1, int reg2, | ||||
|     ts->type = TCG_TYPE_I32; | ||||
|     ts->fixed_reg = 1; | ||||
|     ts->reg = reg2; | ||||
|     ts->val_type = TEMP_VAL_REG; | ||||
|     pstrcpy(buf, sizeof(buf), name); | ||||
|     pstrcat(buf, sizeof(buf), "_1"); | ||||
|     ts->name = strdup(buf); | ||||
| @ -370,7 +370,6 @@ TCGv tcg_global_mem_new(TCGType type, int reg, tcg_target_long offset, | ||||
| #else | ||||
|         ts->mem_offset = offset; | ||||
| #endif | ||||
|         ts->val_type = TEMP_VAL_MEM; | ||||
|         pstrcpy(buf, sizeof(buf), name); | ||||
|         pstrcat(buf, sizeof(buf), "_0"); | ||||
|         ts->name = strdup(buf); | ||||
| @ -386,7 +385,6 @@ TCGv tcg_global_mem_new(TCGType type, int reg, tcg_target_long offset, | ||||
| #else | ||||
|         ts->mem_offset = offset + 4; | ||||
| #endif | ||||
|         ts->val_type = TEMP_VAL_MEM; | ||||
|         pstrcpy(buf, sizeof(buf), name); | ||||
|         pstrcat(buf, sizeof(buf), "_1"); | ||||
|         ts->name = strdup(buf); | ||||
| @ -403,7 +401,6 @@ TCGv tcg_global_mem_new(TCGType type, int reg, tcg_target_long offset, | ||||
|         ts->mem_allocated = 1; | ||||
|         ts->mem_reg = reg; | ||||
|         ts->mem_offset = offset; | ||||
|         ts->val_type = TEMP_VAL_MEM; | ||||
|         ts->name = name; | ||||
|         s->nb_globals++; | ||||
|     } | ||||
| @ -416,90 +413,75 @@ TCGv tcg_temp_new(TCGType type) | ||||
|     TCGTemp *ts; | ||||
|     int idx; | ||||
| 
 | ||||
|     idx = s->nb_temps; | ||||
|     idx = s->first_free_temp[type]; | ||||
|     if (idx != -1) { | ||||
|         /* There is already an available temp with the
 | ||||
|            right type */ | ||||
|         ts = &s->temps[idx]; | ||||
|         s->first_free_temp[type] = ts->next_free_temp; | ||||
|         ts->temp_allocated = 1; | ||||
|     } else { | ||||
|         idx = s->nb_temps; | ||||
| #if TCG_TARGET_REG_BITS == 32 | ||||
|     if (type == TCG_TYPE_I64) { | ||||
|         tcg_temp_alloc(s, s->nb_temps + 1); | ||||
|         ts = &s->temps[s->nb_temps]; | ||||
|         ts->base_type = type; | ||||
|         ts->type = TCG_TYPE_I32; | ||||
|         ts->fixed_reg = 0; | ||||
|         ts->val_type = TEMP_VAL_DEAD; | ||||
|         ts->mem_allocated = 0; | ||||
|         ts->name = NULL; | ||||
|         ts++; | ||||
|         ts->base_type = TCG_TYPE_I32; | ||||
|         ts->type = TCG_TYPE_I32; | ||||
|         ts->val_type = TEMP_VAL_DEAD; | ||||
|         ts->fixed_reg = 0; | ||||
|         ts->mem_allocated = 0; | ||||
|         ts->name = NULL; | ||||
|         s->nb_temps += 2; | ||||
|     } else | ||||
|         if (type == TCG_TYPE_I64) { | ||||
|             tcg_temp_alloc(s, s->nb_temps + 1); | ||||
|             ts = &s->temps[s->nb_temps]; | ||||
|             ts->base_type = type; | ||||
|             ts->type = TCG_TYPE_I32; | ||||
|             ts->temp_allocated = 1; | ||||
|             ts->name = NULL; | ||||
|             ts++; | ||||
|             ts->base_type = TCG_TYPE_I32; | ||||
|             ts->type = TCG_TYPE_I32; | ||||
|             ts->temp_allocated = 1; | ||||
|             ts->name = NULL; | ||||
|             s->nb_temps += 2; | ||||
|         } else | ||||
| #endif | ||||
|     { | ||||
|         tcg_temp_alloc(s, s->nb_temps + 1); | ||||
|         ts = &s->temps[s->nb_temps]; | ||||
|         ts->base_type = type; | ||||
|         ts->type = type; | ||||
|         ts->fixed_reg = 0; | ||||
|         ts->val_type = TEMP_VAL_DEAD; | ||||
|         ts->mem_allocated = 0; | ||||
|         ts->name = NULL; | ||||
|         s->nb_temps++; | ||||
|         { | ||||
|             tcg_temp_alloc(s, s->nb_temps + 1); | ||||
|             ts = &s->temps[s->nb_temps]; | ||||
|             ts->base_type = type; | ||||
|             ts->type = type; | ||||
|             ts->temp_allocated = 1; | ||||
|             ts->name = NULL; | ||||
|             s->nb_temps++; | ||||
|         } | ||||
|     } | ||||
|     return MAKE_TCGV(idx); | ||||
| } | ||||
| 
 | ||||
| TCGv tcg_const_i32(int32_t val) | ||||
| void tcg_temp_free(TCGv arg) | ||||
| { | ||||
|     TCGContext *s = &tcg_ctx; | ||||
|     TCGTemp *ts; | ||||
|     int idx; | ||||
|     int idx = GET_TCGV(arg); | ||||
|     TCGType type; | ||||
| 
 | ||||
|     idx = s->nb_temps; | ||||
|     tcg_temp_alloc(s, idx + 1); | ||||
|     assert(idx >= s->nb_globals && idx < s->nb_temps); | ||||
|     ts = &s->temps[idx]; | ||||
|     ts->base_type = ts->type = TCG_TYPE_I32; | ||||
|     ts->val_type = TEMP_VAL_CONST; | ||||
|     ts->name = NULL; | ||||
|     ts->val = val; | ||||
|     s->nb_temps++; | ||||
|     return MAKE_TCGV(idx); | ||||
|     assert(ts->temp_allocated != 0); | ||||
|     ts->temp_allocated = 0; | ||||
|     type = ts->base_type; | ||||
|     ts->next_free_temp = s->first_free_temp[type]; | ||||
|     s->first_free_temp[type] = idx; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| TCGv tcg_const_i32(int32_t val) | ||||
| { | ||||
|     TCGv t0; | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I32); | ||||
|     tcg_gen_movi_i32(t0, val); | ||||
|     return t0; | ||||
| } | ||||
| 
 | ||||
| TCGv tcg_const_i64(int64_t val) | ||||
| { | ||||
|     TCGContext *s = &tcg_ctx; | ||||
|     TCGTemp *ts; | ||||
|     int idx; | ||||
| 
 | ||||
|     idx = s->nb_temps; | ||||
| #if TCG_TARGET_REG_BITS == 32 | ||||
|     tcg_temp_alloc(s, idx + 2); | ||||
|     ts = &s->temps[idx]; | ||||
|     ts->base_type = TCG_TYPE_I64; | ||||
|     ts->type = TCG_TYPE_I32; | ||||
|     ts->val_type = TEMP_VAL_CONST; | ||||
|     ts->name = NULL; | ||||
|     ts->val = val; | ||||
|     ts++; | ||||
|     ts->base_type = TCG_TYPE_I32; | ||||
|     ts->type = TCG_TYPE_I32; | ||||
|     ts->val_type = TEMP_VAL_CONST; | ||||
|     ts->name = NULL; | ||||
|     ts->val = val >> 32; | ||||
|     s->nb_temps += 2; | ||||
| #else | ||||
|     tcg_temp_alloc(s, idx + 1); | ||||
|     ts = &s->temps[idx]; | ||||
|     ts->base_type = ts->type = TCG_TYPE_I64; | ||||
|     ts->val_type = TEMP_VAL_CONST; | ||||
|     ts->name = NULL; | ||||
|     ts->val = val; | ||||
|     s->nb_temps++; | ||||
| #endif     | ||||
|     return MAKE_TCGV(idx); | ||||
|     TCGv t0; | ||||
|     t0 = tcg_temp_new(TCG_TYPE_I64); | ||||
|     tcg_gen_movi_i64(t0, val); | ||||
|     return t0; | ||||
| } | ||||
| 
 | ||||
| void tcg_register_helper(void *func, const char *name) | ||||
| @ -663,6 +645,8 @@ void tcg_gen_shifti_i64(TCGv ret, TCGv arg1, | ||||
|             tcg_gen_or_i32(TCGV_HIGH(ret), TCGV_HIGH(ret), t0); | ||||
|             tcg_gen_mov_i32(ret, t1); | ||||
|         } | ||||
|         tcg_temp_free(t0); | ||||
|         tcg_temp_free(t1); | ||||
|     } | ||||
| } | ||||
| #endif | ||||
| @ -679,6 +663,12 @@ void tcg_reg_alloc_start(TCGContext *s) | ||||
|             ts->val_type = TEMP_VAL_MEM; | ||||
|         } | ||||
|     } | ||||
|     for(i = s->nb_globals; i < s->nb_temps; i++) { | ||||
|         ts = &s->temps[i]; | ||||
|         ts->val_type = TEMP_VAL_DEAD; | ||||
|         ts->mem_allocated = 0; | ||||
|         ts->fixed_reg = 0; | ||||
|     } | ||||
|     for(i = 0; i < TCG_TARGET_NB_REGS; i++) { | ||||
|         s->reg_to_temp[i] = -1; | ||||
|     } | ||||
| @ -693,11 +683,7 @@ static char *tcg_get_arg_str_idx(TCGContext *s, char *buf, int buf_size, | ||||
|     if (idx < s->nb_globals) { | ||||
|         pstrcpy(buf, buf_size, ts->name); | ||||
|     } else { | ||||
|         if (ts->val_type == TEMP_VAL_CONST) { | ||||
|             snprintf(buf, buf_size, "$0x%" TCG_PRIlx , ts->val); | ||||
|         } else { | ||||
|             snprintf(buf, buf_size, "tmp%d", idx - s->nb_globals); | ||||
|         } | ||||
|         snprintf(buf, buf_size, "tmp%d", idx - s->nb_globals); | ||||
|     } | ||||
|     return buf; | ||||
| } | ||||
| @ -707,37 +693,49 @@ char *tcg_get_arg_str(TCGContext *s, char *buf, int buf_size, TCGv arg) | ||||
|     return tcg_get_arg_str_idx(s, buf, buf_size, GET_TCGV(arg)); | ||||
| } | ||||
| 
 | ||||
| /* find helper definition (XXX: inefficient) */ | ||||
| static int helper_cmp(const void *p1, const void *p2) | ||||
| { | ||||
|     const TCGHelperInfo *th1 = p1; | ||||
|     const TCGHelperInfo *th2 = p2; | ||||
|     if (th1->func < th2->func) | ||||
|         return -1; | ||||
|     else if (th1->func == th2->func) | ||||
|         return 0; | ||||
|     else | ||||
|         return 1; | ||||
| } | ||||
| 
 | ||||
| /* find helper definition (Note: A hash table would be better) */ | ||||
| static TCGHelperInfo *tcg_find_helper(TCGContext *s, tcg_target_ulong val) | ||||
| { | ||||
|     int i; | ||||
|     for(i = 0; i < s->nb_helpers; i++) { | ||||
|         if (s->helpers[i].func == val)  | ||||
|             return &s->helpers[i]; | ||||
|     int m, m_min, m_max; | ||||
|     TCGHelperInfo *th; | ||||
|     tcg_target_ulong v; | ||||
| 
 | ||||
|     if (unlikely(!s->helpers_sorted)) { | ||||
|         qsort(s->helpers, s->nb_helpers, sizeof(TCGHelperInfo),  | ||||
|               helper_cmp); | ||||
|         s->helpers_sorted = 1; | ||||
|     } | ||||
| 
 | ||||
|     /* binary search */ | ||||
|     m_min = 0; | ||||
|     m_max = s->nb_helpers - 1; | ||||
|     while (m_min <= m_max) { | ||||
|         m = (m_min + m_max) >> 1; | ||||
|         th = &s->helpers[m]; | ||||
|         v = th->func; | ||||
|         if (v == val) | ||||
|             return th; | ||||
|         else if (val < v) { | ||||
|             m_max = m - 1; | ||||
|         } else { | ||||
|             m_min = m + 1; | ||||
|         } | ||||
|     } | ||||
|     return NULL; | ||||
| } | ||||
| 
 | ||||
| static const char *tcg_get_helper_str_idx(TCGContext *s, char *buf, int buf_size, | ||||
|                                           int idx) | ||||
| { | ||||
|     TCGTemp *ts; | ||||
|     TCGHelperInfo *th; | ||||
| 
 | ||||
|     ts = &s->temps[idx]; | ||||
|     if (ts->val_type == TEMP_VAL_CONST) { | ||||
|         /* find helper name (XXX: inefficient) */ | ||||
|         th = tcg_find_helper(s, ts->val); | ||||
|         if (th) { | ||||
|             pstrcpy(buf, buf_size, "$"); | ||||
|             pstrcat(buf, buf_size, th->name); | ||||
|             return buf; | ||||
|         } | ||||
|     } | ||||
|     return tcg_get_arg_str_idx(s, buf, buf_size, idx); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void tcg_dump_ops(TCGContext *s, FILE *outfile) | ||||
| { | ||||
|     const uint16_t *opc_ptr; | ||||
| @ -780,7 +778,7 @@ void tcg_dump_ops(TCGContext *s, FILE *outfile) | ||||
| 
 | ||||
|             /* function name */ | ||||
|             fprintf(outfile, "%s", | ||||
|                     tcg_get_helper_str_idx(s, buf, sizeof(buf), args[nb_oargs + nb_iargs - 1])); | ||||
|                     tcg_get_arg_str_idx(s, buf, sizeof(buf), args[nb_oargs + nb_iargs - 1])); | ||||
|             /* flags */ | ||||
|             fprintf(outfile, ",$0x%" TCG_PRIlx, | ||||
|                     args[nb_oargs + nb_iargs]); | ||||
| @ -800,6 +798,29 @@ void tcg_dump_ops(TCGContext *s, FILE *outfile) | ||||
|                             tcg_get_arg_str_idx(s, buf, sizeof(buf), args[nb_oargs + i])); | ||||
|                 } | ||||
|             } | ||||
|         } else if (c == INDEX_op_movi_i32  | ||||
| #if TCG_TARGET_REG_BITS == 64 | ||||
|                    || c == INDEX_op_movi_i64 | ||||
| #endif | ||||
|                    ) { | ||||
|             tcg_target_ulong val; | ||||
|             TCGHelperInfo *th; | ||||
| 
 | ||||
|             nb_oargs = def->nb_oargs; | ||||
|             nb_iargs = def->nb_iargs; | ||||
|             nb_cargs = def->nb_cargs; | ||||
|             fprintf(outfile, " %s %s,$", def->name,  | ||||
|                     tcg_get_arg_str_idx(s, buf, sizeof(buf), args[0])); | ||||
|             val = args[1]; | ||||
|             th = tcg_find_helper(s, val); | ||||
|             if (th) { | ||||
|                 fprintf(outfile, th->name); | ||||
|             } else { | ||||
|                 if (c == INDEX_op_movi_i32) | ||||
|                     fprintf(outfile, "0x%x", (uint32_t)val); | ||||
|                 else | ||||
|                     fprintf(outfile, "0x%" PRIx64 , (uint64_t)val); | ||||
|             } | ||||
|         } else { | ||||
|             fprintf(outfile, " %s ", def->name); | ||||
|             if (c == INDEX_op_nopn) { | ||||
| @ -1281,11 +1302,6 @@ static void check_regs(TCGContext *s) | ||||
|                 dump_regs(s); | ||||
|                 tcg_abort(); | ||||
|         } | ||||
|         if (ts->val_type == TEMP_VAL_CONST && k < s->nb_globals) { | ||||
|             printf("constant forbidden in global %s\n", | ||||
|                    tcg_get_arg_str_idx(s, buf, sizeof(buf), k)); | ||||
|             goto fail; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| #endif | ||||
| @ -1351,47 +1367,80 @@ static int tcg_reg_alloc(TCGContext *s, TCGRegSet reg1, TCGRegSet reg2) | ||||
| } | ||||
| 
 | ||||
| /* save globals to their cannonical location and assume they can be
 | ||||
|    modified be the following code. */ | ||||
| static void save_globals(TCGContext *s) | ||||
|    modified be the following code. 'allocated_regs' is used in case a | ||||
|    temporary registers needs to be allocated to store a constant. */ | ||||
| static void save_globals(TCGContext *s, TCGRegSet allocated_regs) | ||||
| { | ||||
|     TCGTemp *ts; | ||||
|     int i; | ||||
|     int i, reg; | ||||
| 
 | ||||
|     for(i = 0; i < s->nb_globals; i++) { | ||||
|         ts = &s->temps[i]; | ||||
|         if (!ts->fixed_reg) { | ||||
|             if (ts->val_type == TEMP_VAL_REG) { | ||||
|             switch(ts->val_type) { | ||||
|             case TEMP_VAL_REG: | ||||
|                 tcg_reg_free(s, ts->reg); | ||||
|             } else if (ts->val_type == TEMP_VAL_DEAD) { | ||||
|                 break; | ||||
|             case TEMP_VAL_DEAD: | ||||
|                 ts->val_type = TEMP_VAL_MEM; | ||||
|                 break; | ||||
|             case TEMP_VAL_CONST: | ||||
|                 reg = tcg_reg_alloc(s, tcg_target_available_regs[ts->type],  | ||||
|                                     allocated_regs); | ||||
|                 tcg_out_movi(s, ts->type, reg, ts->val); | ||||
|                 tcg_out_st(s, ts->type, reg, ts->mem_reg, ts->mem_offset); | ||||
|                 ts->val_type = TEMP_VAL_MEM; | ||||
|                 break; | ||||
|             case TEMP_VAL_MEM: | ||||
|                 break; | ||||
|             default: | ||||
|                 tcg_abort(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /* at the end of a basic block, we assume all temporaries are dead and
 | ||||
|    all globals are stored at their canonical location */ | ||||
| /* XXX: optimize by handling constants in another array ? */ | ||||
| void tcg_reg_alloc_bb_end(TCGContext *s) | ||||
|    all globals are stored at their canonical location. */ | ||||
| static void tcg_reg_alloc_bb_end(TCGContext *s, TCGRegSet allocated_regs) | ||||
| { | ||||
|     TCGTemp *ts; | ||||
|     int i; | ||||
| 
 | ||||
|     save_globals(s); | ||||
| 
 | ||||
|     for(i = s->nb_globals; i < s->nb_temps; i++) { | ||||
|         ts = &s->temps[i]; | ||||
|         if (ts->val_type != TEMP_VAL_CONST) { | ||||
|             if (ts->val_type == TEMP_VAL_REG) { | ||||
|                 s->reg_to_temp[ts->reg] = -1; | ||||
|             } | ||||
|             ts->val_type = TEMP_VAL_DEAD; | ||||
|         if (ts->val_type == TEMP_VAL_REG) { | ||||
|             s->reg_to_temp[ts->reg] = -1; | ||||
|         } | ||||
|         ts->val_type = TEMP_VAL_DEAD; | ||||
|     } | ||||
| 
 | ||||
|     save_globals(s, allocated_regs); | ||||
| } | ||||
| 
 | ||||
| #define IS_DEAD_IARG(n) ((dead_iargs >> (n)) & 1) | ||||
| 
 | ||||
| static void tcg_reg_alloc_movi(TCGContext *s, const TCGArg *args) | ||||
| { | ||||
|     TCGTemp *ots; | ||||
|     tcg_target_ulong val; | ||||
| 
 | ||||
|     ots = &s->temps[args[0]]; | ||||
|     val = args[1]; | ||||
| 
 | ||||
|     if (ots->fixed_reg) { | ||||
|         /* for fixed registers, we do not do any constant
 | ||||
|            propagation */ | ||||
|         tcg_out_movi(s, ots->type, ots->reg, val); | ||||
|     } else { | ||||
|         /* The movi is not explicitely generated here */ | ||||
|         if (ots->val_type == TEMP_VAL_REG) | ||||
|             s->reg_to_temp[ots->reg] = -1; | ||||
|         ots->val_type = TEMP_VAL_CONST; | ||||
|         ots->val = val; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| static void tcg_reg_alloc_mov(TCGContext *s, const TCGOpDef *def, | ||||
|                               const TCGArg *args, | ||||
|                               unsigned int dead_iargs) | ||||
| @ -1404,6 +1453,7 @@ static void tcg_reg_alloc_mov(TCGContext *s, const TCGOpDef *def, | ||||
|     ts = &s->temps[args[1]]; | ||||
|     arg_ct = &def->args_ct[0]; | ||||
| 
 | ||||
|     /* XXX: always mark arg dead if IS_DEAD_IARG(0) */ | ||||
|     if (ts->val_type == TEMP_VAL_REG) { | ||||
|         if (IS_DEAD_IARG(0) && !ts->fixed_reg && !ots->fixed_reg) { | ||||
|             /* the mov can be suppressed */ | ||||
| @ -1430,12 +1480,17 @@ static void tcg_reg_alloc_mov(TCGContext *s, const TCGOpDef *def, | ||||
|         } | ||||
|         tcg_out_ld(s, ts->type, reg, ts->mem_reg, ts->mem_offset); | ||||
|     } else if (ts->val_type == TEMP_VAL_CONST) { | ||||
|         if (ots->val_type == TEMP_VAL_REG) { | ||||
|         if (ots->fixed_reg) { | ||||
|             reg = ots->reg; | ||||
|             tcg_out_movi(s, ots->type, reg, ts->val); | ||||
|         } else { | ||||
|             reg = tcg_reg_alloc(s, arg_ct->u.regs, s->reserved_regs); | ||||
|             /* propagate constant */ | ||||
|             if (ots->val_type == TEMP_VAL_REG) | ||||
|                 s->reg_to_temp[ots->reg] = -1; | ||||
|             ots->val_type = TEMP_VAL_CONST; | ||||
|             ots->val = ts->val; | ||||
|             return; | ||||
|         } | ||||
|         tcg_out_movi(s, ots->type, reg, ts->val); | ||||
|     } else { | ||||
|         tcg_abort(); | ||||
|     } | ||||
| @ -1487,10 +1542,13 @@ static void tcg_reg_alloc_op(TCGContext *s, | ||||
|                 new_args[i] = ts->val; | ||||
|                 goto iarg_end; | ||||
|             } else { | ||||
|                 /* need to move to a register*/ | ||||
|                 /* need to move to a register */ | ||||
|                 reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|                 tcg_out_movi(s, ts->type, reg, ts->val); | ||||
|                 goto iarg_end1; | ||||
|                 ts->val_type = TEMP_VAL_REG; | ||||
|                 ts->reg = reg; | ||||
|                 ts->mem_coherent = 0; | ||||
|                 s->reg_to_temp[reg] = arg; | ||||
|             } | ||||
|         } | ||||
|         assert(ts->val_type == TEMP_VAL_REG); | ||||
| @ -1518,78 +1576,78 @@ static void tcg_reg_alloc_op(TCGContext *s, | ||||
|             reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|             tcg_out_mov(s, reg, ts->reg); | ||||
|         } | ||||
|     iarg_end1: | ||||
|         new_args[i] = reg; | ||||
|         const_args[i] = 0; | ||||
|         tcg_regset_set_reg(allocated_regs, reg); | ||||
|     iarg_end: ; | ||||
|     } | ||||
|      | ||||
|     /* mark dead temporaries and free the associated registers */ | ||||
|     for(i = 0; i < nb_iargs; i++) { | ||||
|         arg = args[nb_oargs + i]; | ||||
|         if (IS_DEAD_IARG(i)) { | ||||
|     if (def->flags & TCG_OPF_BB_END) { | ||||
|         tcg_reg_alloc_bb_end(s, allocated_regs); | ||||
|     } else { | ||||
|         /* mark dead temporaries and free the associated registers */ | ||||
|         for(i = 0; i < nb_iargs; i++) { | ||||
|             arg = args[nb_oargs + i]; | ||||
|             if (IS_DEAD_IARG(i)) { | ||||
|                 ts = &s->temps[arg]; | ||||
|                 if (!ts->fixed_reg) { | ||||
|                     if (ts->val_type == TEMP_VAL_REG) | ||||
|                         s->reg_to_temp[ts->reg] = -1; | ||||
|                     ts->val_type = TEMP_VAL_DEAD; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         if (def->flags & TCG_OPF_CALL_CLOBBER) { | ||||
|             /* XXX: permit generic clobber register list ? */  | ||||
|             for(reg = 0; reg < TCG_TARGET_NB_REGS; reg++) { | ||||
|                 if (tcg_regset_test_reg(tcg_target_call_clobber_regs, reg)) { | ||||
|                     tcg_reg_free(s, reg); | ||||
|                 } | ||||
|             } | ||||
|             /* XXX: for load/store we could do that only for the slow path
 | ||||
|                (i.e. when a memory callback is called) */ | ||||
|              | ||||
|             /* store globals and free associated registers (we assume the insn
 | ||||
|                can modify any global. */ | ||||
|             save_globals(s, allocated_regs); | ||||
|         } | ||||
|          | ||||
|         /* satisfy the output constraints */ | ||||
|         tcg_regset_set(allocated_regs, s->reserved_regs); | ||||
|         for(k = 0; k < nb_oargs; k++) { | ||||
|             i = def->sorted_args[k]; | ||||
|             arg = args[i]; | ||||
|             arg_ct = &def->args_ct[i]; | ||||
|             ts = &s->temps[arg]; | ||||
|             if (ts->val_type != TEMP_VAL_CONST && !ts->fixed_reg) { | ||||
|             if (arg_ct->ct & TCG_CT_ALIAS) { | ||||
|                 reg = new_args[arg_ct->alias_index]; | ||||
|             } else { | ||||
|                 /* if fixed register, we try to use it */ | ||||
|                 reg = ts->reg; | ||||
|                 if (ts->fixed_reg && | ||||
|                     tcg_regset_test_reg(arg_ct->u.regs, reg)) { | ||||
|                     goto oarg_end; | ||||
|                 } | ||||
|                 reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|             } | ||||
|             tcg_regset_set_reg(allocated_regs, reg); | ||||
|             /* if a fixed register is used, then a move will be done afterwards */ | ||||
|             if (!ts->fixed_reg) { | ||||
|                 if (ts->val_type == TEMP_VAL_REG) | ||||
|                     s->reg_to_temp[ts->reg] = -1; | ||||
|                 ts->val_type = TEMP_VAL_DEAD; | ||||
|                 ts->val_type = TEMP_VAL_REG; | ||||
|                 ts->reg = reg; | ||||
|                 /* temp value is modified, so the value kept in memory is
 | ||||
|                    potentially not the same */ | ||||
|                 ts->mem_coherent = 0;  | ||||
|                 s->reg_to_temp[reg] = arg; | ||||
|             } | ||||
|         oarg_end: | ||||
|             new_args[i] = reg; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (def->flags & TCG_OPF_CALL_CLOBBER) { | ||||
|         /* XXX: permit generic clobber register list ? */  | ||||
|         for(reg = 0; reg < TCG_TARGET_NB_REGS; reg++) { | ||||
|             if (tcg_regset_test_reg(tcg_target_call_clobber_regs, reg)) { | ||||
|                 tcg_reg_free(s, reg); | ||||
|             } | ||||
|         } | ||||
|         /* XXX: for load/store we could do that only for the slow path
 | ||||
|            (i.e. when a memory callback is called) */ | ||||
| 
 | ||||
|         /* store globals and free associated registers (we assume the insn
 | ||||
|            can modify any global. */ | ||||
|         save_globals(s); | ||||
|     } | ||||
| 
 | ||||
|     /* satisfy the output constraints */ | ||||
|     tcg_regset_set(allocated_regs, s->reserved_regs); | ||||
|     for(k = 0; k < nb_oargs; k++) { | ||||
|         i = def->sorted_args[k]; | ||||
|         arg = args[i]; | ||||
|         arg_ct = &def->args_ct[i]; | ||||
|         ts = &s->temps[arg]; | ||||
|         if (arg_ct->ct & TCG_CT_ALIAS) { | ||||
|             reg = new_args[arg_ct->alias_index]; | ||||
|         } else { | ||||
|             /* if fixed register, we try to use it */ | ||||
|             reg = ts->reg; | ||||
|             if (ts->fixed_reg && | ||||
|                 tcg_regset_test_reg(arg_ct->u.regs, reg)) { | ||||
|                 goto oarg_end; | ||||
|             } | ||||
|             reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|         } | ||||
|         tcg_regset_set_reg(allocated_regs, reg); | ||||
|         /* if a fixed register is used, then a move will be done afterwards */ | ||||
|         if (!ts->fixed_reg) { | ||||
|             if (ts->val_type == TEMP_VAL_REG) | ||||
|                 s->reg_to_temp[ts->reg] = -1; | ||||
|             ts->val_type = TEMP_VAL_REG; | ||||
|             ts->reg = reg; | ||||
|             /* temp value is modified, so the value kept in memory is
 | ||||
|                potentially not the same */ | ||||
|             ts->mem_coherent = 0;  | ||||
|             s->reg_to_temp[reg] = arg; | ||||
|         } | ||||
|     oarg_end: | ||||
|         new_args[i] = reg; | ||||
|     } | ||||
| 
 | ||||
|     if (def->flags & TCG_OPF_BB_END) | ||||
|         tcg_reg_alloc_bb_end(s); | ||||
| 
 | ||||
|     /* emit instruction */ | ||||
|     tcg_out_op(s, opc, new_args, const_args); | ||||
|      | ||||
| @ -1708,6 +1766,7 @@ static int tcg_reg_alloc_call(TCGContext *s, const TCGOpDef *def, | ||||
|         reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|         tcg_out_ld(s, ts->type, reg, ts->mem_reg, ts->mem_offset); | ||||
|         func_arg = reg; | ||||
|         tcg_regset_set_reg(allocated_regs, reg); | ||||
|     } else if (ts->val_type == TEMP_VAL_REG) { | ||||
|         reg = ts->reg; | ||||
|         if (!tcg_regset_test_reg(arg_ct->u.regs, reg)) { | ||||
| @ -1715,6 +1774,7 @@ static int tcg_reg_alloc_call(TCGContext *s, const TCGOpDef *def, | ||||
|             tcg_out_mov(s, reg, ts->reg); | ||||
|         } | ||||
|         func_arg = reg; | ||||
|         tcg_regset_set_reg(allocated_regs, reg); | ||||
|     } else if (ts->val_type == TEMP_VAL_CONST) { | ||||
|         if (tcg_target_const_match(func_addr, arg_ct)) { | ||||
|             const_func_arg = 1; | ||||
| @ -1723,17 +1783,19 @@ static int tcg_reg_alloc_call(TCGContext *s, const TCGOpDef *def, | ||||
|             reg = tcg_reg_alloc(s, arg_ct->u.regs, allocated_regs); | ||||
|             tcg_out_movi(s, ts->type, reg, func_addr); | ||||
|             func_arg = reg; | ||||
|             tcg_regset_set_reg(allocated_regs, reg); | ||||
|         } | ||||
|     } else { | ||||
|         tcg_abort(); | ||||
|     } | ||||
|          | ||||
|      | ||||
|     /* mark dead temporaries and free the associated registers */ | ||||
|     for(i = 0; i < nb_iargs; i++) { | ||||
|         arg = args[nb_oargs + i]; | ||||
|         if (IS_DEAD_IARG(i)) { | ||||
|             ts = &s->temps[arg]; | ||||
|             if (ts->val_type != TEMP_VAL_CONST && !ts->fixed_reg) { | ||||
|             if (!ts->fixed_reg) { | ||||
|                 if (ts->val_type == TEMP_VAL_REG) | ||||
|                     s->reg_to_temp[ts->reg] = -1; | ||||
|                 ts->val_type = TEMP_VAL_DEAD; | ||||
| @ -1750,7 +1812,7 @@ static int tcg_reg_alloc_call(TCGContext *s, const TCGOpDef *def, | ||||
|      | ||||
|     /* store globals and free associated registers (we assume the call
 | ||||
|        can modify any global. */ | ||||
|     save_globals(s); | ||||
|     save_globals(s, allocated_regs); | ||||
| 
 | ||||
|     tcg_out_op(s, opc, &func_arg, &const_func_arg); | ||||
|      | ||||
| @ -1763,7 +1825,7 @@ static int tcg_reg_alloc_call(TCGContext *s, const TCGOpDef *def, | ||||
|         arg = args[i]; | ||||
|         ts = &s->temps[arg]; | ||||
|         reg = tcg_target_call_oarg_regs[i]; | ||||
|         tcg_reg_free(s, reg); | ||||
|         assert(s->reg_to_temp[reg] == -1); | ||||
|         if (ts->fixed_reg) { | ||||
|             if (ts->reg != reg) { | ||||
|                 tcg_out_mov(s, ts->reg, reg); | ||||
| @ -1863,6 +1925,12 @@ static inline int tcg_gen_code_common(TCGContext *s, uint8_t *gen_code_buf, | ||||
|             dead_iargs = s->op_dead_iargs[op_index]; | ||||
|             tcg_reg_alloc_mov(s, def, args, dead_iargs); | ||||
|             break; | ||||
|         case INDEX_op_movi_i32: | ||||
| #if TCG_TARGET_REG_BITS == 64 | ||||
|         case INDEX_op_movi_i64: | ||||
| #endif | ||||
|             tcg_reg_alloc_movi(s, args); | ||||
|             break; | ||||
|         case INDEX_op_debug_insn_start: | ||||
|             /* debug instruction */ | ||||
|             break; | ||||
| @ -1879,7 +1947,7 @@ static inline int tcg_gen_code_common(TCGContext *s, uint8_t *gen_code_buf, | ||||
|                 TCGTemp *ts; | ||||
|                 ts = &s->temps[args[0]]; | ||||
|                 /* mark the temporary as dead */ | ||||
|                 if (ts->val_type != TEMP_VAL_CONST && !ts->fixed_reg) { | ||||
|                 if (!ts->fixed_reg) { | ||||
|                     if (ts->val_type == TEMP_VAL_REG) | ||||
|                         s->reg_to_temp[ts->reg] = -1; | ||||
|                     ts->val_type = TEMP_VAL_DEAD; | ||||
| @ -1900,7 +1968,7 @@ static inline int tcg_gen_code_common(TCGContext *s, uint8_t *gen_code_buf, | ||||
|             /* must never happen here */ | ||||
|             tcg_abort(); | ||||
|         case INDEX_op_set_label: | ||||
|             tcg_reg_alloc_bb_end(s); | ||||
|             tcg_reg_alloc_bb_end(s, s->reserved_regs); | ||||
|             tcg_out_label(s, args[0], (long)s->code_ptr); | ||||
|             break; | ||||
|         case INDEX_op_call: | ||||
| @ -1916,7 +1984,7 @@ static inline int tcg_gen_code_common(TCGContext *s, uint8_t *gen_code_buf, | ||||
| #ifdef CONFIG_PROFILER | ||||
|             s->old_op_count++; | ||||
| #endif | ||||
|             tcg_reg_alloc_bb_end(s); | ||||
|             tcg_reg_alloc_bb_end(s, s->reserved_regs); | ||||
|             if (search_pc >= 0) { | ||||
|                 s->code_ptr += def->copy_size; | ||||
|                 args += def->nb_args; | ||||
|  | ||||
							
								
								
									
										11
									
								
								tcg/tcg.h
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								tcg/tcg.h
									
									
									
									
									
								
							| @ -98,6 +98,7 @@ typedef int TCGType; | ||||
| 
 | ||||
| #define TCG_TYPE_I32 0 | ||||
| #define TCG_TYPE_I64 1 | ||||
| #define TCG_TYPE_COUNT 2 /* number of different types */ | ||||
| 
 | ||||
| #if TCG_TARGET_REG_BITS == 32 | ||||
| #define TCG_TYPE_PTR TCG_TYPE_I32 | ||||
| @ -188,6 +189,9 @@ typedef struct TCGTemp { | ||||
|     unsigned int fixed_reg:1; | ||||
|     unsigned int mem_coherent:1; | ||||
|     unsigned int mem_allocated:1; | ||||
|     unsigned int temp_allocated:1; /* never used for code gen */ | ||||
|     /* index of next free temp of same base type, -1 if end */ | ||||
|     int next_free_temp; | ||||
|     const char *name; | ||||
| } TCGTemp; | ||||
| 
 | ||||
| @ -208,6 +212,8 @@ struct TCGContext { | ||||
|     TCGTemp *temps; /* globals first, temps after */ | ||||
|     int nb_globals; | ||||
|     int nb_temps; | ||||
|     int first_free_temp[TCG_TYPE_COUNT]; /* index of free temps, -1 if none */ | ||||
| 
 | ||||
|     /* constant indexes (end of temp array) */ | ||||
|     int const_start; | ||||
|     int const_end; | ||||
| @ -236,6 +242,7 @@ struct TCGContext { | ||||
|     TCGHelperInfo *helpers; | ||||
|     int nb_helpers; | ||||
|     int allocated_helpers; | ||||
|     int helpers_sorted; | ||||
| 
 | ||||
| #ifdef CONFIG_PROFILER | ||||
|     /* profiling info */ | ||||
| @ -299,6 +306,7 @@ TCGv tcg_global_reg2_new_hack(TCGType type, int reg1, int reg2, | ||||
| TCGv tcg_global_mem_new(TCGType type, int reg, tcg_target_long offset, | ||||
|                         const char *name); | ||||
| TCGv tcg_temp_new(TCGType type); | ||||
| void tcg_temp_free(TCGv arg); | ||||
| char *tcg_get_arg_str(TCGContext *s, char *buf, int buf_size, TCGv arg); | ||||
| void tcg_dump_info(FILE *f, | ||||
|                    int (*cpu_fprintf)(FILE *f, const char *fmt, ...)); | ||||
| @ -381,9 +389,6 @@ TCGv tcg_const_i64(int64_t val); | ||||
| 
 | ||||
| void tcg_out_reloc(TCGContext *s, uint8_t *code_ptr, int type,  | ||||
|                    int label_index, long addend); | ||||
| void tcg_reg_alloc_start(TCGContext *s); | ||||
| void tcg_reg_alloc_bb_end(TCGContext *s); | ||||
| void tcg_liveness_analysis(TCGContext *s); | ||||
| const TCGArg *tcg_gen_code_op(TCGContext *s, int opc, const TCGArg *args1, | ||||
|                               unsigned int dead_iargs); | ||||
| 
 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 bellard
						bellard