qemu-s390x
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: [PATCH 9/9] target/s390x: Use Int128 for passing float128


From: Ilya Leoshkevich
Subject: Re: [PATCH 9/9] target/s390x: Use Int128 for passing float128
Date: Wed, 2 Nov 2022 10:38:48 +0100

On Fri, Oct 21, 2022 at 05:30:06PM +1000, Richard Henderson wrote:
> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
> ---
>  target/s390x/helper.h          | 32 ++++++-------
>  target/s390x/tcg/fpu_helper.c  | 88 ++++++++++++++--------------------
>  target/s390x/tcg/translate.c   | 76 ++++++++++++++++++++---------
>  target/s390x/tcg/insn-data.def | 30 ++++++------
>  4 files changed, 121 insertions(+), 105 deletions(-)
> 
> diff --git a/target/s390x/helper.h b/target/s390x/helper.h
> index 429131a85e..481b9019f9 100644
> --- a/target/s390x/helper.h
> +++ b/target/s390x/helper.h
> @@ -41,55 +41,55 @@ DEF_HELPER_4(csst, i32, env, i32, i64, i64)
>  DEF_HELPER_4(csst_parallel, i32, env, i32, i64, i64)
>  DEF_HELPER_FLAGS_3(aeb, TCG_CALL_NO_WG, i64, env, i64, i64)
>  DEF_HELPER_FLAGS_3(adb, TCG_CALL_NO_WG, i64, env, i64, i64)
> -DEF_HELPER_FLAGS_5(axb, TCG_CALL_NO_WG, i128, env, i64, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(axb, TCG_CALL_NO_WG, i128, env, i128, i128)
>  DEF_HELPER_FLAGS_3(seb, TCG_CALL_NO_WG, i64, env, i64, i64)
>  DEF_HELPER_FLAGS_3(sdb, TCG_CALL_NO_WG, i64, env, i64, i64)
> -DEF_HELPER_FLAGS_5(sxb, TCG_CALL_NO_WG, i128, env, i64, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(sxb, TCG_CALL_NO_WG, i128, env, i128, i128)
>  DEF_HELPER_FLAGS_3(deb, TCG_CALL_NO_WG, i64, env, i64, i64)
>  DEF_HELPER_FLAGS_3(ddb, TCG_CALL_NO_WG, i64, env, i64, i64)
> -DEF_HELPER_FLAGS_5(dxb, TCG_CALL_NO_WG, i128, env, i64, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(dxb, TCG_CALL_NO_WG, i128, env, i128, i128)
>  DEF_HELPER_FLAGS_3(meeb, TCG_CALL_NO_WG, i64, env, i64, i64)
>  DEF_HELPER_FLAGS_3(mdeb, TCG_CALL_NO_WG, i64, env, i64, i64)
>  DEF_HELPER_FLAGS_3(mdb, TCG_CALL_NO_WG, i64, env, i64, i64)
> -DEF_HELPER_FLAGS_5(mxb, TCG_CALL_NO_WG, i128, env, i64, i64, i64, i64)
> -DEF_HELPER_FLAGS_4(mxdb, TCG_CALL_NO_WG, i128, env, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(mxb, TCG_CALL_NO_WG, i128, env, i128, i128)
> +DEF_HELPER_FLAGS_3(mxdb, TCG_CALL_NO_WG, i128, env, i128, i64)
>  DEF_HELPER_FLAGS_2(ldeb, TCG_CALL_NO_WG, i64, env, i64)
> -DEF_HELPER_FLAGS_4(ldxb, TCG_CALL_NO_WG, i64, env, i64, i64, i32)
> +DEF_HELPER_FLAGS_3(ldxb, TCG_CALL_NO_WG, i64, env, i128, i32)
>  DEF_HELPER_FLAGS_2(lxdb, TCG_CALL_NO_WG, i128, env, i64)
>  DEF_HELPER_FLAGS_2(lxeb, TCG_CALL_NO_WG, i128, env, i64)
>  DEF_HELPER_FLAGS_3(ledb, TCG_CALL_NO_WG, i64, env, i64, i32)
> -DEF_HELPER_FLAGS_4(lexb, TCG_CALL_NO_WG, i64, env, i64, i64, i32)
> +DEF_HELPER_FLAGS_3(lexb, TCG_CALL_NO_WG, i64, env, i128, i32)
>  DEF_HELPER_FLAGS_3(ceb, TCG_CALL_NO_WG_SE, i32, env, i64, i64)
>  DEF_HELPER_FLAGS_3(cdb, TCG_CALL_NO_WG_SE, i32, env, i64, i64)
> -DEF_HELPER_FLAGS_5(cxb, TCG_CALL_NO_WG_SE, i32, env, i64, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(cxb, TCG_CALL_NO_WG_SE, i32, env, i128, i128)
>  DEF_HELPER_FLAGS_3(keb, TCG_CALL_NO_WG, i32, env, i64, i64)
>  DEF_HELPER_FLAGS_3(kdb, TCG_CALL_NO_WG, i32, env, i64, i64)
> -DEF_HELPER_FLAGS_5(kxb, TCG_CALL_NO_WG, i32, env, i64, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(kxb, TCG_CALL_NO_WG, i32, env, i128, i128)
>  DEF_HELPER_3(cgeb, i64, env, i64, i32)
>  DEF_HELPER_3(cgdb, i64, env, i64, i32)
> -DEF_HELPER_4(cgxb, i64, env, i64, i64, i32)
> +DEF_HELPER_3(cgxb, i64, env, i128, i32)
>  DEF_HELPER_3(cfeb, i64, env, i64, i32)
>  DEF_HELPER_3(cfdb, i64, env, i64, i32)
> -DEF_HELPER_4(cfxb, i64, env, i64, i64, i32)
> +DEF_HELPER_3(cfxb, i64, env, i128, i32)
>  DEF_HELPER_3(clgeb, i64, env, i64, i32)
>  DEF_HELPER_3(clgdb, i64, env, i64, i32)
> -DEF_HELPER_4(clgxb, i64, env, i64, i64, i32)
> +DEF_HELPER_3(clgxb, i64, env, i128, i32)
>  DEF_HELPER_3(clfeb, i64, env, i64, i32)
>  DEF_HELPER_3(clfdb, i64, env, i64, i32)
> -DEF_HELPER_4(clfxb, i64, env, i64, i64, i32)
> +DEF_HELPER_3(clfxb, i64, env, i128, i32)
>  DEF_HELPER_FLAGS_3(fieb, TCG_CALL_NO_WG, i64, env, i64, i32)
>  DEF_HELPER_FLAGS_3(fidb, TCG_CALL_NO_WG, i64, env, i64, i32)
> -DEF_HELPER_FLAGS_4(fixb, TCG_CALL_NO_WG, i128, env, i64, i64, i32)
> +DEF_HELPER_FLAGS_3(fixb, TCG_CALL_NO_WG, i128, env, i128, i32)
>  DEF_HELPER_FLAGS_4(maeb, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
>  DEF_HELPER_FLAGS_4(madb, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
>  DEF_HELPER_FLAGS_4(mseb, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
>  DEF_HELPER_FLAGS_4(msdb, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
>  DEF_HELPER_FLAGS_3(tceb, TCG_CALL_NO_RWG_SE, i32, env, i64, i64)
>  DEF_HELPER_FLAGS_3(tcdb, TCG_CALL_NO_RWG_SE, i32, env, i64, i64)
> -DEF_HELPER_FLAGS_4(tcxb, TCG_CALL_NO_RWG_SE, i32, env, i64, i64, i64)
> +DEF_HELPER_FLAGS_3(tcxb, TCG_CALL_NO_RWG_SE, i32, env, i128, i64)
>  DEF_HELPER_FLAGS_2(sqeb, TCG_CALL_NO_WG, i64, env, i64)
>  DEF_HELPER_FLAGS_2(sqdb, TCG_CALL_NO_WG, i64, env, i64)
> -DEF_HELPER_FLAGS_3(sqxb, TCG_CALL_NO_WG, i128, env, i64, i64)
> +DEF_HELPER_FLAGS_2(sqxb, TCG_CALL_NO_WG, i128, env, i128)
>  DEF_HELPER_FLAGS_1(cvd, TCG_CALL_NO_RWG_SE, i64, s32)
>  DEF_HELPER_FLAGS_4(pack, TCG_CALL_NO_WG, void, env, i32, i64, i64)
>  DEF_HELPER_FLAGS_4(pka, TCG_CALL_NO_WG, void, env, i64, i64, i32)
> diff --git a/target/s390x/tcg/fpu_helper.c b/target/s390x/tcg/fpu_helper.c
> index a584794be6..5a322e3f87 100644
> --- a/target/s390x/tcg/fpu_helper.c
> +++ b/target/s390x/tcg/fpu_helper.c
> @@ -39,6 +39,11 @@ static inline Int128 RET128(float128 f)
>      return int128_make128(f.low, f.high);
>  }
>  
> +static inline float128 ARG128(Int128 i)
> +{
> +    return make_float128(int128_gethi(i), int128_getlo(i));
> +}
> +
>  uint8_t s390_softfloat_exc_to_ieee(unsigned int exc)
>  {
>      uint8_t s390_exc = 0;
> @@ -227,12 +232,9 @@ uint64_t HELPER(adb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP addition */
> -Int128 HELPER(axb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +Int128 HELPER(axb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    float128 ret = float128_add(make_float128(ah, al),
> -                                make_float128(bh, bl),
> -                                &env->fpu_status);
> +    float128 ret = float128_add(ARG128(a), ARG128(b), &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
> @@ -254,12 +256,9 @@ uint64_t HELPER(sdb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP subtraction */
> -Int128 HELPER(sxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +Int128 HELPER(sxb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    float128 ret = float128_sub(make_float128(ah, al),
> -                                make_float128(bh, bl),
> -                                &env->fpu_status);
> +    float128 ret = float128_sub(ARG128(a), ARG128(b), &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
> @@ -281,12 +280,9 @@ uint64_t HELPER(ddb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP division */
> -Int128 HELPER(dxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +Int128 HELPER(dxb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    float128 ret = float128_div(make_float128(ah, al),
> -                                make_float128(bh, bl),
> -                                &env->fpu_status);
> +    float128 ret = float128_div(ARG128(a), ARG128(b), &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
> @@ -317,21 +313,18 @@ uint64_t HELPER(mdeb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP multiplication */
> -Int128 HELPER(mxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +Int128 HELPER(mxb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    float128 ret = float128_mul(make_float128(ah, al),
> -                                make_float128(bh, bl),
> -                                &env->fpu_status);
> +    float128 ret = float128_mul(ARG128(a), ARG128(b), &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
>  
>  /* 128/64-bit FP multiplication */
> -Int128 HELPER(mxdb)(CPUS390XState *env, uint64_t ah, uint64_t al, uint64_t 
> f2)
> +Int128 HELPER(mxdb)(CPUS390XState *env, Int128 a, uint64_t f2)
>  {
>      float128 ret = float64_to_float128(f2, &env->fpu_status);
> -    ret = float128_mul(make_float128(ah, al), ret, &env->fpu_status);
> +    ret = float128_mul(ARG128(a), ret, &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
> @@ -345,11 +338,10 @@ uint64_t HELPER(ldeb)(CPUS390XState *env, uint64_t f2)
>  }
>  
>  /* convert 128-bit float to 64-bit float */
> -uint64_t HELPER(ldxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                      uint32_t m34)
> +uint64_t HELPER(ldxb)(CPUS390XState *env, Int128 a, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float64 ret = float128_to_float64(make_float128(ah, al), 
> &env->fpu_status);
> +    float64 ret = float128_to_float64(ARG128(a), &env->fpu_status);
>  
>      s390_restore_bfp_rounding_mode(env, old_mode);
>      handle_exceptions(env, xxc_from_m34(m34), GETPC());
> @@ -384,11 +376,10 @@ uint64_t HELPER(ledb)(CPUS390XState *env, uint64_t f2, 
> uint32_t m34)
>  }
>  
>  /* convert 128-bit float to 32-bit float */
> -uint64_t HELPER(lexb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                      uint32_t m34)
> +uint64_t HELPER(lexb)(CPUS390XState *env, Int128 a, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float32 ret = float128_to_float32(make_float128(ah, al), 
> &env->fpu_status);
> +    float32 ret = float128_to_float32(ARG128(a), &env->fpu_status);
>  
>      s390_restore_bfp_rounding_mode(env, old_mode);
>      handle_exceptions(env, xxc_from_m34(m34), GETPC());
> @@ -412,11 +403,9 @@ uint32_t HELPER(cdb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP compare */
> -uint32_t HELPER(cxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +uint32_t HELPER(cxb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    FloatRelation cmp = float128_compare_quiet(make_float128(ah, al),
> -                                               make_float128(bh, bl),
> +    FloatRelation cmp = float128_compare_quiet(ARG128(a), ARG128(b),
>                                                 &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return float_comp_to_cc(env, cmp);
> @@ -564,10 +553,10 @@ uint64_t HELPER(cgdb)(CPUS390XState *env, uint64_t v2, 
> uint32_t m34)
>  }
>  
>  /* convert 128-bit float to 64-bit int */
> -uint64_t HELPER(cgxb)(CPUS390XState *env, uint64_t h, uint64_t l, uint32_t 
> m34)
> +uint64_t HELPER(cgxb)(CPUS390XState *env, Int128 i2, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float128 v2 = make_float128(h, l);
> +    float128 v2 = ARG128(i2);
>      int64_t ret = float128_to_int64(v2, &env->fpu_status);
>      uint32_t cc = set_cc_conv_f128(v2, &env->fpu_status);
>  
> @@ -613,10 +602,10 @@ uint64_t HELPER(cfdb)(CPUS390XState *env, uint64_t v2, 
> uint32_t m34)
>  }
>  
>  /* convert 128-bit float to 32-bit int */
> -uint64_t HELPER(cfxb)(CPUS390XState *env, uint64_t h, uint64_t l, uint32_t 
> m34)
> +uint64_t HELPER(cfxb)(CPUS390XState *env, Int128 i2, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float128 v2 = make_float128(h, l);
> +    float128 v2 = ARG128(i2);
>      int32_t ret = float128_to_int32(v2, &env->fpu_status);
>      uint32_t cc = set_cc_conv_f128(v2, &env->fpu_status);
>  
> @@ -662,10 +651,10 @@ uint64_t HELPER(clgdb)(CPUS390XState *env, uint64_t v2, 
> uint32_t m34)
>  }
>  
>  /* convert 128-bit float to 64-bit uint */
> -uint64_t HELPER(clgxb)(CPUS390XState *env, uint64_t h, uint64_t l, uint32_t 
> m34)
> +uint64_t HELPER(clgxb)(CPUS390XState *env, Int128 i2, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float128 v2 = make_float128(h, l);
> +    float128 v2 = ARG128(i2);
>      uint64_t ret = float128_to_uint64(v2, &env->fpu_status);
>      uint32_t cc = set_cc_conv_f128(v2, &env->fpu_status);
>  
> @@ -711,10 +700,10 @@ uint64_t HELPER(clfdb)(CPUS390XState *env, uint64_t v2, 
> uint32_t m34)
>  }
>  
>  /* convert 128-bit float to 32-bit uint */
> -uint64_t HELPER(clfxb)(CPUS390XState *env, uint64_t h, uint64_t l, uint32_t 
> m34)
> +uint64_t HELPER(clfxb)(CPUS390XState *env, Int128 i2, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float128 v2 = make_float128(h, l);
> +    float128 v2 = ARG128(i2);
>      uint32_t ret = float128_to_uint32(v2, &env->fpu_status);
>      uint32_t cc = set_cc_conv_f128(v2, &env->fpu_status);
>  
> @@ -750,11 +739,10 @@ uint64_t HELPER(fidb)(CPUS390XState *env, uint64_t f2, 
> uint32_t m34)
>  }
>  
>  /* round to integer 128-bit */
> -Int128 HELPER(fixb)(CPUS390XState *env, uint64_t ah, uint64_t al, uint32_t 
> m34)
> +Int128 HELPER(fixb)(CPUS390XState *env, Int128 a, uint32_t m34)
>  {
>      int old_mode = s390_swap_bfp_rounding_mode(env, round_from_m34(m34));
> -    float128 ret = float128_round_to_int(make_float128(ah, al),
> -                                         &env->fpu_status);
> +    float128 ret = float128_round_to_int(ARG128(a), &env->fpu_status);
>  
>      s390_restore_bfp_rounding_mode(env, old_mode);
>      handle_exceptions(env, xxc_from_m34(m34), GETPC());
> @@ -778,11 +766,9 @@ uint32_t HELPER(kdb)(CPUS390XState *env, uint64_t f1, 
> uint64_t f2)
>  }
>  
>  /* 128-bit FP compare and signal */
> -uint32_t HELPER(kxb)(CPUS390XState *env, uint64_t ah, uint64_t al,
> -                     uint64_t bh, uint64_t bl)
> +uint32_t HELPER(kxb)(CPUS390XState *env, Int128 a, Int128 b)
>  {
> -    FloatRelation cmp = float128_compare(make_float128(ah, al),
> -                                         make_float128(bh, bl),
> +    FloatRelation cmp = float128_compare(ARG128(a), ARG128(b),
>                                           &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return float_comp_to_cc(env, cmp);
> @@ -869,9 +855,9 @@ uint32_t HELPER(tcdb)(CPUS390XState *env, uint64_t v1, 
> uint64_t m2)
>  }
>  
>  /* test data class 128-bit */
> -uint32_t HELPER(tcxb)(CPUS390XState *env, uint64_t ah, uint64_t al, uint64_t 
> m2)
> +uint32_t HELPER(tcxb)(CPUS390XState *env, Int128 a, uint64_t m2)
>  {
> -    return (m2 & float128_dcmask(env, make_float128(ah, al))) != 0;
> +    return (m2 & float128_dcmask(env, ARG128(a))) != 0;
>  }
>  
>  /* square root 32-bit */
> @@ -891,9 +877,9 @@ uint64_t HELPER(sqdb)(CPUS390XState *env, uint64_t f2)
>  }
>  
>  /* square root 128-bit */
> -Int128 HELPER(sqxb)(CPUS390XState *env, uint64_t ah, uint64_t al)
> +Int128 HELPER(sqxb)(CPUS390XState *env, Int128 a)
>  {
> -    float128 ret = float128_sqrt(make_float128(ah, al), &env->fpu_status);
> +    float128 ret = float128_sqrt(ARG128(a), &env->fpu_status);
>      handle_exceptions(env, false, GETPC());
>      return RET128(ret);
>  }
> diff --git a/target/s390x/tcg/translate.c b/target/s390x/tcg/translate.c
> index d1ffbb8710..8023bbab2f 100644
> --- a/target/s390x/tcg/translate.c
> +++ b/target/s390x/tcg/translate.c
> @@ -305,6 +305,18 @@ static TCGv_i64 load_freg32_i64(int reg)
>      return r;
>  }
>  
> +static TCGv_i128 load_freg_128(int reg)
> +{
> +    TCGv_i64 h = load_freg(reg);
> +    TCGv_i64 l = load_freg(reg + 2);
> +    TCGv_i128 r = tcg_temp_new_i128();
> +
> +    tcg_gen_concat_i64_i128(r, l, h);
> +    tcg_temp_free_i64(h);
> +    tcg_temp_free_i64(l);
> +    return r;
> +}
> +
>  static void store_reg(int reg, TCGv_i64 v)
>  {
>      tcg_gen_mov_i64(regs[reg], v);
> @@ -1103,7 +1115,7 @@ typedef struct {
>      bool g_out, g_out2, g_in1, g_in2;
>      TCGv_i64 out, out2, in1, in2;
>      TCGv_i64 addr1;
> -    TCGv_i128 out_128;
> +    TCGv_i128 out_128, in1_128, in2_128;
>  } DisasOps;
>  
>  /* Instructions can place constraints on their operands, raising 
> specification
> @@ -1462,7 +1474,7 @@ static DisasJumpType op_adb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_axb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_axb(o->out_128, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_axb(o->out_128, cpu_env, o->in1_128, o->in2_128);
>      return DISAS_NEXT;
>  }
>  
> @@ -1778,7 +1790,7 @@ static DisasJumpType op_cdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_cxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_cxb(cc_op, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_cxb(cc_op, cpu_env, o->in1_128, o->in2_128);
>      set_cc_static(s);
>      return DISAS_NEXT;
>  }
> @@ -1841,7 +1853,7 @@ static DisasJumpType op_cfxb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_cfxb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_cfxb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      set_cc_static(s);
>      return DISAS_NEXT;
> @@ -1880,7 +1892,7 @@ static DisasJumpType op_cgxb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_cgxb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_cgxb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      set_cc_static(s);
>      return DISAS_NEXT;
> @@ -1919,7 +1931,7 @@ static DisasJumpType op_clfxb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_clfxb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_clfxb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      set_cc_static(s);
>      return DISAS_NEXT;
> @@ -1958,7 +1970,7 @@ static DisasJumpType op_clgxb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_clgxb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_clgxb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      set_cc_static(s);
>      return DISAS_NEXT;
> @@ -2448,7 +2460,7 @@ static DisasJumpType op_ddb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_dxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_dxb(o->out_128, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_dxb(o->out_128, cpu_env, o->in1_128, o->in2_128);
>      return DISAS_NEXT;
>  }
>  
> @@ -2553,7 +2565,7 @@ static DisasJumpType op_fixb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_fixb(o->out_128, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_fixb(o->out_128, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      return DISAS_NEXT;
>  }
> @@ -2772,7 +2784,7 @@ static DisasJumpType op_kdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_kxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_kxb(cc_op, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_kxb(cc_op, cpu_env, o->in1_128, o->in2_128);
>      set_cc_static(s);
>      return DISAS_NEXT;
>  }
> @@ -2846,7 +2858,7 @@ static DisasJumpType op_ldxb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_ldxb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_ldxb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      return DISAS_NEXT;
>  }
> @@ -2858,7 +2870,7 @@ static DisasJumpType op_lexb(DisasContext *s, DisasOps 
> *o)
>      if (!m34) {
>          return DISAS_NORETURN;
>      }
> -    gen_helper_lexb(o->out, cpu_env, o->in1, o->in2, m34);
> +    gen_helper_lexb(o->out, cpu_env, o->in2_128, m34);
>      tcg_temp_free_i32(m34);
>      return DISAS_NEXT;
>  }
> @@ -3586,13 +3598,13 @@ static DisasJumpType op_mdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_mxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_mxb(o->out_128, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_mxb(o->out_128, cpu_env, o->in1_128, o->in2_128);
>      return DISAS_NEXT;
>  }
>  
>  static DisasJumpType op_mxdb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_mxdb(o->out_128, cpu_env, o->out, o->out2, o->in2);
> +    gen_helper_mxdb(o->out_128, cpu_env, o->in1_128, o->in2);
>      return DISAS_NEXT;
>  }
>  
> @@ -4057,7 +4069,7 @@ static DisasJumpType op_sdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_sxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_sxb(o->out_128, cpu_env, o->out, o->out2, o->in1, o->in2);
> +    gen_helper_sxb(o->out_128, cpu_env, o->in1_128, o->in2_128);
>      return DISAS_NEXT;
>  }
>  
> @@ -4075,7 +4087,7 @@ static DisasJumpType op_sqdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_sqxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_sqxb(o->out_128, cpu_env, o->in1, o->in2);
> +    gen_helper_sqxb(o->out_128, cpu_env, o->in2_128);
>      return DISAS_NEXT;
>  }
>  
> @@ -4854,7 +4866,7 @@ static DisasJumpType op_tcdb(DisasContext *s, DisasOps 
> *o)
>  
>  static DisasJumpType op_tcxb(DisasContext *s, DisasOps *o)
>  {
> -    gen_helper_tcxb(cc_op, cpu_env, o->out, o->out2, o->in2);
> +    gen_helper_tcxb(cc_op, cpu_env, o->in1_128, o->in2);
>      set_cc_static(s);
>      return DISAS_NEXT;
>  }
> @@ -5389,8 +5401,6 @@ static void prep_new_P(DisasContext *s, DisasOps *o)
>  
>  static void prep_new_x(DisasContext *s, DisasOps *o)
>  {
> -    o->out = tcg_temp_new_i64();
> -    o->out2 = tcg_temp_new_i64();
>      o->out_128 = tcg_temp_new_i128();
>  }
>  #define SPEC_prep_new_x 0
> @@ -5413,10 +5423,7 @@ static void prep_r1_P(DisasContext *s, DisasOps *o)
>  
>  static void prep_x1(DisasContext *s, DisasOps *o)
>  {
> -    o->out = load_freg(get_field(s, r1));
> -    o->out2 = load_freg(get_field(s, r1) + 2);
> -    o->out_128 = tcg_temp_new_i128();
> -    tcg_gen_concat_i64_i128(o->out_128, o->out2, o->out);
> +    o->out_128 = load_freg_128(get_field(s, r1));
>  }
>  #define SPEC_prep_x1 SPEC_r1_f128
>  
> @@ -5515,6 +5522,11 @@ static void wout_x1(DisasContext *s, DisasOps *o)
>  {
>      int f1 = get_field(s, r1);
>  
> +    /* Split out_128 into out+out2 for cout_f128. */
> +    tcg_debug_assert(o->out == NULL);
> +    o->out = tcg_temp_new_i64();
> +    o->out2 = tcg_temp_new_i64();
> +
>      tcg_gen_extr_i128_i64(o->out2, o->out, o->out_128);
>      store_freg(f1, o->out);
>      store_freg(f1 + 2, o->out2);
> @@ -5757,6 +5769,12 @@ static void in1_f1(DisasContext *s, DisasOps *o)
>  }
>  #define SPEC_in1_f1 0
>  
> +static void in1_x1(DisasContext *s, DisasOps *o)
> +{
> +    o->in1_128 = load_freg_128(get_field(s, r1));
> +}
> +#define SPEC_in1_x1 SPEC_r2_f128
> +
>  /* Load the high double word of an extended (128-bit) format FP number */
>  static void in1_x2h(DisasContext *s, DisasOps *o)
>  {
> @@ -5966,6 +5984,12 @@ static void in2_f2(DisasContext *s, DisasOps *o)
>  }
>  #define SPEC_in2_f2 0
>  
> +static void in2_x2(DisasContext *s, DisasOps *o)
> +{
> +    o->in2_128 = load_freg_128(get_field(s, r2));
> +}
> +#define SPEC_in2_x2 SPEC_r2_f128
> +
>  /* Load the low double word of an extended (128-bit) format FP number */
>  static void in2_x2l(DisasContext *s, DisasOps *o)
>  {
> @@ -6588,6 +6612,12 @@ static DisasJumpType translate_one(CPUS390XState *env, 
> DisasContext *s)
>      if (o.out_128) {
>          tcg_temp_free_i128(o.out_128);
>      }
> +    if (o.in1_128) {
> +        tcg_temp_free_i128(o.in1_128);
> +    }
> +    if (o.in2_128) {
> +        tcg_temp_free_i128(o.in2_128);
> +    }
>      /* io should be the last instruction in tb when icount is enabled */
>      if (unlikely(icount && ret == DISAS_NEXT)) {
>          ret = DISAS_TOO_MANY;
> diff --git a/target/s390x/tcg/insn-data.def b/target/s390x/tcg/insn-data.def
> index 20bf20c766..26523746d6 100644
> --- a/target/s390x/tcg/insn-data.def
> +++ b/target/s390x/tcg/insn-data.def
> @@ -34,7 +34,7 @@
>      C(0xe318, AGF,     RXY_a, Z,   r1, m2_32s, r1, 0, add, adds64)
>      F(0xb30a, AEBR,    RRE,   Z,   e1, e2, new, e1, aeb, f32, IF_BFP)
>      F(0xb31a, ADBR,    RRE,   Z,   f1, f2, new, f1, adb, f64, IF_BFP)
> -    F(0xb34a, AXBR,    RRE,   Z,   x2h, x2l, x1, x1, axb, f128, IF_BFP)
> +    F(0xb34a, AXBR,    RRE,   Z,   x1, x2, new_x, x1, axb, f128, IF_BFP)
>      F(0xed0a, AEB,     RXE,   Z,   e1, m2_32u, new, e1, aeb, f32, IF_BFP)
>      F(0xed1a, ADB,     RXE,   Z,   f1, m2_64, new, f1, adb, f64, IF_BFP)
>  /* ADD HIGH */
> @@ -172,13 +172,13 @@
>      C(0xe330, CGF,     RXY_a, Z,   r1_o, m2_32s, 0, 0, 0, cmps64)
>      F(0xb309, CEBR,    RRE,   Z,   e1, e2, 0, 0, ceb, 0, IF_BFP)
>      F(0xb319, CDBR,    RRE,   Z,   f1, f2, 0, 0, cdb, 0, IF_BFP)
> -    F(0xb349, CXBR,    RRE,   Z,   x2h, x2l, x1, 0, cxb, 0, IF_BFP)
> +    F(0xb349, CXBR,    RRE,   Z,   x1, x2, 0, 0, cxb, 0, IF_BFP)
>      F(0xed09, CEB,     RXE,   Z,   e1, m2_32u, 0, 0, ceb, 0, IF_BFP)
>      F(0xed19, CDB,     RXE,   Z,   f1, m2_64, 0, 0, cdb, 0, IF_BFP)
>  /* COMPARE AND SIGNAL */
>      F(0xb308, KEBR,    RRE,   Z,   e1, e2, 0, 0, keb, 0, IF_BFP)
>      F(0xb318, KDBR,    RRE,   Z,   f1, f2, 0, 0, kdb, 0, IF_BFP)
> -    F(0xb348, KXBR,    RRE,   Z,   x2h, x2l, x1, 0, kxb, 0, IF_BFP)
> +    F(0xb348, KXBR,    RRE,   Z,   x1, x2, 0, 0, kxb, 0, IF_BFP)
>      F(0xed08, KEB,     RXE,   Z,   e1, m2_32u, 0, 0, keb, 0, IF_BFP)
>      F(0xed18, KDB,     RXE,   Z,   f1, m2_64, 0, 0, kdb, 0, IF_BFP)
>  /* COMPARE IMMEDIATE */
> @@ -299,10 +299,10 @@
>  /* CONVERT TO FIXED */
>      F(0xb398, CFEBR,   RRF_e, Z,   0, e2, new, r1_32, cfeb, 0, IF_BFP)
>      F(0xb399, CFDBR,   RRF_e, Z,   0, f2, new, r1_32, cfdb, 0, IF_BFP)
> -    F(0xb39a, CFXBR,   RRF_e, Z,   x2h, x2l, new, r1_32, cfxb, 0, IF_BFP)
> +    F(0xb39a, CFXBR,   RRF_e, Z,   0, x2, new, r1_32, cfxb, 0, IF_BFP)
>      F(0xb3a8, CGEBR,   RRF_e, Z,   0, e2, r1, 0, cgeb, 0, IF_BFP)
>      F(0xb3a9, CGDBR,   RRF_e, Z,   0, f2, r1, 0, cgdb, 0, IF_BFP)
> -    F(0xb3aa, CGXBR,   RRF_e, Z,   x2h, x2l, r1, 0, cgxb, 0, IF_BFP)
> +    F(0xb3aa, CGXBR,   RRF_e, Z,   0, x2, r1, 0, cgxb, 0, IF_BFP)
>  /* CONVERT FROM FIXED */
>      F(0xb394, CEFBR,   RRF_e, Z,   0, r2_32s, new, e1, cegb, 0, IF_BFP)
>      F(0xb395, CDFBR,   RRF_e, Z,   0, r2_32s, new, f1, cdgb, 0, IF_BFP)
> @@ -313,10 +313,10 @@
>  /* CONVERT TO LOGICAL */
>      F(0xb39c, CLFEBR,  RRF_e, FPE, 0, e2, new, r1_32, clfeb, 0, IF_BFP)
>      F(0xb39d, CLFDBR,  RRF_e, FPE, 0, f2, new, r1_32, clfdb, 0, IF_BFP)
> -    F(0xb39e, CLFXBR,  RRF_e, FPE, x2h, x2l, new, r1_32, clfxb, 0, IF_BFP)
> +    F(0xb39e, CLFXBR,  RRF_e, FPE, 0, x2, new, r1_32, clfxb, 0, IF_BFP)
>      F(0xb3ac, CLGEBR,  RRF_e, FPE, 0, e2, r1, 0, clgeb, 0, IF_BFP)
>      F(0xb3ad, CLGDBR,  RRF_e, FPE, 0, f2, r1, 0, clgdb, 0, IF_BFP)
> -    F(0xb3ae, CLGXBR,  RRF_e, FPE, x2h, x2l, r1, 0, clgxb, 0, IF_BFP)
> +    F(0xb3ae, CLGXBR,  RRF_e, FPE, 0, x2, r1, 0, clgxb, 0, IF_BFP)
>  /* CONVERT FROM LOGICAL */
>      F(0xb390, CELFBR,  RRF_e, FPE, 0, r2_32u, new, e1, celgb, 0, IF_BFP)
>      F(0xb391, CDLFBR,  RRF_e, FPE, 0, r2_32u, new, f1, cdlgb, 0, IF_BFP)
> @@ -343,7 +343,7 @@
>      C(0x5d00, D,       RX_a,  Z,   r1_D32, m2_32s, new_P, r1_P32, divs32, 0)
>      F(0xb30d, DEBR,    RRE,   Z,   e1, e2, new, e1, deb, 0, IF_BFP)
>      F(0xb31d, DDBR,    RRE,   Z,   f1, f2, new, f1, ddb, 0, IF_BFP)
> -    F(0xb34d, DXBR,    RRE,   Z,   x2h, x2l, x1, x1, dxb, 0, IF_BFP)
> +    F(0xb34d, DXBR,    RRE,   Z,   x1, x2, new_x, x1, dxb, 0, IF_BFP)
>      F(0xed0d, DEB,     RXE,   Z,   e1, m2_32u, new, e1, deb, 0, IF_BFP)
>      F(0xed1d, DDB,     RXE,   Z,   f1, m2_64, new, f1, ddb, 0, IF_BFP)
>  /* DIVIDE LOGICAL */
> @@ -597,7 +597,7 @@
>  /* LOAD FP INTEGER */
>      F(0xb357, FIEBR,   RRF_e, Z,   0, e2, new, e1, fieb, 0, IF_BFP)
>      F(0xb35f, FIDBR,   RRF_e, Z,   0, f2, new, f1, fidb, 0, IF_BFP)
> -    F(0xb347, FIXBR,   RRF_e, Z,   x2h, x2l, new_x, x1, fixb, 0, IF_BFP)
> +    F(0xb347, FIXBR,   RRF_e, Z,   0, x2, new_x, x1, fixb, 0, IF_BFP)
>  
>  /* LOAD LENGTHENED */
>      F(0xb304, LDEBR,   RRE,   Z,   0, e2, new, f1, ldeb, 0, IF_BFP)
> @@ -610,8 +610,8 @@
>      F(0xed24, LDE,     RXE,   Z,   0, m2_32u, new, f1, lde, 0, IF_AFP1)
>  /* LOAD ROUNDED */
>      F(0xb344, LEDBR,   RRF_e, Z,   0, f2, new, e1, ledb, 0, IF_BFP)
> -    F(0xb345, LDXBR,   RRF_e, Z,   x2h, x2l, new, f1, ldxb, 0, IF_BFP)
> -    F(0xb346, LEXBR,   RRF_e, Z,   x2h, x2l, new, e1, lexb, 0, IF_BFP)
> +    F(0xb345, LDXBR,   RRF_e, Z,   0, x2, new, f1, ldxb, 0, IF_BFP)
> +    F(0xb346, LEXBR,   RRF_e, Z,   0, x2, new, e1, lexb, 0, IF_BFP)
>  
>  /* LOAD MULTIPLE */
>      C(0x9800, LM,      RS_a,  Z,   0, a2, 0, 0, lm32, 0)
> @@ -666,7 +666,7 @@
>      C(0xe384, MG,      RXY_a, MIE2,r1p1_o, m2_64, r1_P, 0, muls128, 0)
>      F(0xb317, MEEBR,   RRE,   Z,   e1, e2, new, e1, meeb, 0, IF_BFP)
>      F(0xb31c, MDBR,    RRE,   Z,   f1, f2, new, f1, mdb, 0, IF_BFP)
> -    F(0xb34c, MXBR,    RRE,   Z,   x2h, x2l, x1, x1, mxb, 0, IF_BFP)
> +    F(0xb34c, MXBR,    RRE,   Z,   x1, x2, new_x, x1, mxb, 0, IF_BFP)
>      F(0xb30c, MDEBR,   RRE,   Z,   f1, e2, new, f1, mdeb, 0, IF_BFP)
>      F(0xb307, MXDBR,   RRE,   Z,   0, f2, x1, x1, mxdb, 0, IF_BFP)
>      F(0xed17, MEEB,    RXE,   Z,   e1, m2_32u, new, e1, meeb, 0, IF_BFP)
> @@ -835,7 +835,7 @@
>  /* SQUARE ROOT */
>      F(0xb314, SQEBR,   RRE,   Z,   0, e2, new, e1, sqeb, 0, IF_BFP)
>      F(0xb315, SQDBR,   RRE,   Z,   0, f2, new, f1, sqdb, 0, IF_BFP)
> -    F(0xb316, SQXBR,   RRE,   Z,   x2h, x2l, new_x, x1, sqxb, 0, IF_BFP)
> +    F(0xb316, SQXBR,   RRE,   Z,   0, x2, new_x, x1, sqxb, 0, IF_BFP)
>      F(0xed14, SQEB,    RXE,   Z,   0, m2_32u, new, e1, sqeb, 0, IF_BFP)
>      F(0xed15, SQDB,    RXE,   Z,   0, m2_64, new, f1, sqdb, 0, IF_BFP)
>  
> @@ -913,7 +913,7 @@
>      C(0xe319, SGF,     RXY_a, Z,   r1, m2_32s, r1, 0, sub, subs64)
>      F(0xb30b, SEBR,    RRE,   Z,   e1, e2, new, e1, seb, f32, IF_BFP)
>      F(0xb31b, SDBR,    RRE,   Z,   f1, f2, new, f1, sdb, f64, IF_BFP)
> -    F(0xb34b, SXBR,    RRE,   Z,   x2h, x2l, x1, x1, sxb, f128, IF_BFP)
> +    F(0xb34b, SXBR,    RRE,   Z,   x1, x2, new_x, x1, sxb, f128, IF_BFP)
>      F(0xed0b, SEB,     RXE,   Z,   e1, m2_32u, new, e1, seb, f32, IF_BFP)
>      F(0xed1b, SDB,     RXE,   Z,   f1, m2_64, new, f1, sdb, f64, IF_BFP)
>  /* SUBTRACT HALFWORD */
> @@ -957,7 +957,7 @@
>  /* TEST DATA CLASS */
>      F(0xed10, TCEB,    RXE,   Z,   e1, a2, 0, 0, tceb, 0, IF_BFP)
>      F(0xed11, TCDB,    RXE,   Z,   f1, a2, 0, 0, tcdb, 0, IF_BFP)
> -    F(0xed12, TCXB,    RXE,   Z,   0, a2, x1, 0, tcxb, 0, IF_BFP)
> +    F(0xed12, TCXB,    RXE,   Z,   x1, a2, 0, 0, tcxb, 0, IF_BFP)
>  
>  /* TEST DECIMAL */
>      C(0xebc0, TP,      RSL,   E2,  la1, 0, 0, 0, tp, 0)
> -- 
> 2.34.1
> 
> 

Hi,

I ran valgrind's testsuite with this patch, and their bpf-4 test
triggered an assertion in the

    (insn->spec & SPEC_r2_f128 && !is_fp_pair(get_field(s, r2)))

condition. The following fixup helped:


--- a/target/s390x/tcg/translate.c
+++ b/target/s390x/tcg/translate.c
@@ -5771,14 +5771,14 @@ static void in1_x1(DisasContext *s, DisasOps *o)
 {
     o->in1_128 = load_freg_128(get_field(s, r1));
 }
-#define SPEC_in1_x1 SPEC_r2_f128
+#define SPEC_in1_x1 SPEC_r1_f128
 
 /* Load the high double word of an extended (128-bit) format FP number */
 static void in1_x2h(DisasContext *s, DisasOps *o)
 {
     o->in1 = load_freg(get_field(s, r2));
 }
-#define SPEC_in1_x2h SPEC_r2_f128
+#define SPEC_in1_x2h SPEC_r1_f128
 
 static void in1_f3(DisasContext *s, DisasOps *o)
 {


Best regards,
Ilya



reply via email to

[Prev in Thread] Current Thread [Next in Thread]