1/* 2 * QEMU TCG support -- s390x vector instruction translation functions 3 * 4 * Copyright (C) 2019 Red Hat Inc 5 * 6 * Authors: 7 * David Hildenbrand <david@redhat.com> 8 * 9 * This work is licensed under the terms of the GNU GPL, version 2 or later. 10 * See the COPYING file in the top-level directory. 11 */ 12 13/* 14 * For most instructions that use the same element size for reads and 15 * writes, we can use real gvec vector expansion, which potantially uses 16 * real host vector instructions. As they only work up to 64 bit elements, 17 * 128 bit elements (vector is a single element) have to be handled 18 * differently. Operations that are too complicated to encode via TCG ops 19 * are handled via gvec ool (out-of-line) handlers. 20 * 21 * As soon as instructions use different element sizes for reads and writes 22 * or access elements "out of their element scope" we expand them manually 23 * in fancy loops, as gvec expansion does not deal with actual element 24 * numbers and does also not support access to other elements. 25 * 26 * 128 bit elements: 27 * As we only have i32/i64, such elements have to be loaded into two 28 * i64 values and can then be processed e.g. by tcg_gen_add2_i64. 29 * 30 * Sizes: 31 * On s390x, the operand size (oprsz) and the maximum size (maxsz) are 32 * always 16 (128 bit). What gvec code calls "vece", s390x calls "es", 33 * a.k.a. "element size". These values nicely map to MO_8 ... MO_64. Only 34 * 128 bit element size has to be treated in a special way (MO_64 + 1). 35 * We will use ES_* instead of MO_* for this reason in this file. 36 * 37 * CC handling: 38 * As gvec ool-helpers can currently not return values (besides via 39 * pointers like vectors or cpu_env), whenever we have to set the CC and 40 * can't conclude the value from the result vector, we will directly 41 * set it in "env->cc_op" and mark it as static via set_cc_static()". 42 * Whenever this is done, the helper writes globals (cc_op). 43 */ 44 45#define NUM_VEC_ELEMENT_BYTES(es) (1 << (es)) 46#define NUM_VEC_ELEMENTS(es) (16 / NUM_VEC_ELEMENT_BYTES(es)) 47#define NUM_VEC_ELEMENT_BITS(es) (NUM_VEC_ELEMENT_BYTES(es) * BITS_PER_BYTE) 48 49#define ES_8 MO_8 50#define ES_16 MO_16 51#define ES_32 MO_32 52#define ES_64 MO_64 53#define ES_128 4 54 55/* Floating-Point Format */ 56#define FPF_SHORT 2 57#define FPF_LONG 3 58#define FPF_EXT 4 59 60static inline bool valid_vec_element(uint8_t enr, MemOp es) 61{ 62 return !(enr & ~(NUM_VEC_ELEMENTS(es) - 1)); 63} 64 65static void read_vec_element_i64(TCGv_i64 dst, uint8_t reg, uint8_t enr, 66 MemOp memop) 67{ 68 const int offs = vec_reg_offset(reg, enr, memop & MO_SIZE); 69 70 switch ((unsigned)memop) { 71 case ES_8: 72 tcg_gen_ld8u_i64(dst, cpu_env, offs); 73 break; 74 case ES_16: 75 tcg_gen_ld16u_i64(dst, cpu_env, offs); 76 break; 77 case ES_32: 78 tcg_gen_ld32u_i64(dst, cpu_env, offs); 79 break; 80 case ES_8 | MO_SIGN: 81 tcg_gen_ld8s_i64(dst, cpu_env, offs); 82 break; 83 case ES_16 | MO_SIGN: 84 tcg_gen_ld16s_i64(dst, cpu_env, offs); 85 break; 86 case ES_32 | MO_SIGN: 87 tcg_gen_ld32s_i64(dst, cpu_env, offs); 88 break; 89 case ES_64: 90 case ES_64 | MO_SIGN: 91 tcg_gen_ld_i64(dst, cpu_env, offs); 92 break; 93 default: 94 g_assert_not_reached(); 95 } 96} 97 98static void read_vec_element_i32(TCGv_i32 dst, uint8_t reg, uint8_t enr, 99 MemOp memop) 100{ 101 const int offs = vec_reg_offset(reg, enr, memop & MO_SIZE); 102 103 switch (memop) { 104 case ES_8: 105 tcg_gen_ld8u_i32(dst, cpu_env, offs); 106 break; 107 case ES_16: 108 tcg_gen_ld16u_i32(dst, cpu_env, offs); 109 break; 110 case ES_8 | MO_SIGN: 111 tcg_gen_ld8s_i32(dst, cpu_env, offs); 112 break; 113 case ES_16 | MO_SIGN: 114 tcg_gen_ld16s_i32(dst, cpu_env, offs); 115 break; 116 case ES_32: 117 case ES_32 | MO_SIGN: 118 tcg_gen_ld_i32(dst, cpu_env, offs); 119 break; 120 default: 121 g_assert_not_reached(); 122 } 123} 124 125static void write_vec_element_i64(TCGv_i64 src, int reg, uint8_t enr, 126 MemOp memop) 127{ 128 const int offs = vec_reg_offset(reg, enr, memop & MO_SIZE); 129 130 switch (memop) { 131 case ES_8: 132 tcg_gen_st8_i64(src, cpu_env, offs); 133 break; 134 case ES_16: 135 tcg_gen_st16_i64(src, cpu_env, offs); 136 break; 137 case ES_32: 138 tcg_gen_st32_i64(src, cpu_env, offs); 139 break; 140 case ES_64: 141 tcg_gen_st_i64(src, cpu_env, offs); 142 break; 143 default: 144 g_assert_not_reached(); 145 } 146} 147 148static void write_vec_element_i32(TCGv_i32 src, int reg, uint8_t enr, 149 MemOp memop) 150{ 151 const int offs = vec_reg_offset(reg, enr, memop & MO_SIZE); 152 153 switch (memop) { 154 case ES_8: 155 tcg_gen_st8_i32(src, cpu_env, offs); 156 break; 157 case ES_16: 158 tcg_gen_st16_i32(src, cpu_env, offs); 159 break; 160 case ES_32: 161 tcg_gen_st_i32(src, cpu_env, offs); 162 break; 163 default: 164 g_assert_not_reached(); 165 } 166} 167 168static void get_vec_element_ptr_i64(TCGv_ptr ptr, uint8_t reg, TCGv_i64 enr, 169 uint8_t es) 170{ 171 TCGv_i64 tmp = tcg_temp_new_i64(); 172 173 /* mask off invalid parts from the element nr */ 174 tcg_gen_andi_i64(tmp, enr, NUM_VEC_ELEMENTS(es) - 1); 175 176 /* convert it to an element offset relative to cpu_env (vec_reg_offset() */ 177 tcg_gen_shli_i64(tmp, tmp, es); 178#if !HOST_BIG_ENDIAN 179 tcg_gen_xori_i64(tmp, tmp, 8 - NUM_VEC_ELEMENT_BYTES(es)); 180#endif 181 tcg_gen_addi_i64(tmp, tmp, vec_full_reg_offset(reg)); 182 183 /* generate the final ptr by adding cpu_env */ 184 tcg_gen_trunc_i64_ptr(ptr, tmp); 185 tcg_gen_add_ptr(ptr, ptr, cpu_env); 186 187 tcg_temp_free_i64(tmp); 188} 189 190#define gen_gvec_2(v1, v2, gen) \ 191 tcg_gen_gvec_2(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 192 16, 16, gen) 193#define gen_gvec_2s(v1, v2, c, gen) \ 194 tcg_gen_gvec_2s(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 195 16, 16, c, gen) 196#define gen_gvec_2_ool(v1, v2, data, fn) \ 197 tcg_gen_gvec_2_ool(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 198 16, 16, data, fn) 199#define gen_gvec_2i_ool(v1, v2, c, data, fn) \ 200 tcg_gen_gvec_2i_ool(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 201 c, 16, 16, data, fn) 202#define gen_gvec_2_ptr(v1, v2, ptr, data, fn) \ 203 tcg_gen_gvec_2_ptr(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 204 ptr, 16, 16, data, fn) 205#define gen_gvec_3(v1, v2, v3, gen) \ 206 tcg_gen_gvec_3(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 207 vec_full_reg_offset(v3), 16, 16, gen) 208#define gen_gvec_3_ool(v1, v2, v3, data, fn) \ 209 tcg_gen_gvec_3_ool(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 210 vec_full_reg_offset(v3), 16, 16, data, fn) 211#define gen_gvec_3_ptr(v1, v2, v3, ptr, data, fn) \ 212 tcg_gen_gvec_3_ptr(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 213 vec_full_reg_offset(v3), ptr, 16, 16, data, fn) 214#define gen_gvec_3i(v1, v2, v3, c, gen) \ 215 tcg_gen_gvec_3i(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 216 vec_full_reg_offset(v3), 16, 16, c, gen) 217#define gen_gvec_4(v1, v2, v3, v4, gen) \ 218 tcg_gen_gvec_4(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 219 vec_full_reg_offset(v3), vec_full_reg_offset(v4), \ 220 16, 16, gen) 221#define gen_gvec_4_ool(v1, v2, v3, v4, data, fn) \ 222 tcg_gen_gvec_4_ool(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 223 vec_full_reg_offset(v3), vec_full_reg_offset(v4), \ 224 16, 16, data, fn) 225#define gen_gvec_4_ptr(v1, v2, v3, v4, ptr, data, fn) \ 226 tcg_gen_gvec_4_ptr(vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 227 vec_full_reg_offset(v3), vec_full_reg_offset(v4), \ 228 ptr, 16, 16, data, fn) 229#define gen_gvec_dup_i64(es, v1, c) \ 230 tcg_gen_gvec_dup_i64(es, vec_full_reg_offset(v1), 16, 16, c) 231#define gen_gvec_mov(v1, v2) \ 232 tcg_gen_gvec_mov(0, vec_full_reg_offset(v1), vec_full_reg_offset(v2), 16, \ 233 16) 234#define gen_gvec_dup_imm(es, v1, c) \ 235 tcg_gen_gvec_dup_imm(es, vec_full_reg_offset(v1), 16, 16, c); 236#define gen_gvec_fn_2(fn, es, v1, v2) \ 237 tcg_gen_gvec_##fn(es, vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 238 16, 16) 239#define gen_gvec_fn_2i(fn, es, v1, v2, c) \ 240 tcg_gen_gvec_##fn(es, vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 241 c, 16, 16) 242#define gen_gvec_fn_2s(fn, es, v1, v2, s) \ 243 tcg_gen_gvec_##fn(es, vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 244 s, 16, 16) 245#define gen_gvec_fn_3(fn, es, v1, v2, v3) \ 246 tcg_gen_gvec_##fn(es, vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 247 vec_full_reg_offset(v3), 16, 16) 248#define gen_gvec_fn_4(fn, es, v1, v2, v3, v4) \ 249 tcg_gen_gvec_##fn(es, vec_full_reg_offset(v1), vec_full_reg_offset(v2), \ 250 vec_full_reg_offset(v3), vec_full_reg_offset(v4), 16, 16) 251 252/* 253 * Helper to carry out a 128 bit vector computation using 2 i64 values per 254 * vector. 255 */ 256typedef void (*gen_gvec128_3_i64_fn)(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, 257 TCGv_i64 ah, TCGv_i64 bl, TCGv_i64 bh); 258static void gen_gvec128_3_i64(gen_gvec128_3_i64_fn fn, uint8_t d, uint8_t a, 259 uint8_t b) 260{ 261 TCGv_i64 dh = tcg_temp_new_i64(); 262 TCGv_i64 dl = tcg_temp_new_i64(); 263 TCGv_i64 ah = tcg_temp_new_i64(); 264 TCGv_i64 al = tcg_temp_new_i64(); 265 TCGv_i64 bh = tcg_temp_new_i64(); 266 TCGv_i64 bl = tcg_temp_new_i64(); 267 268 read_vec_element_i64(ah, a, 0, ES_64); 269 read_vec_element_i64(al, a, 1, ES_64); 270 read_vec_element_i64(bh, b, 0, ES_64); 271 read_vec_element_i64(bl, b, 1, ES_64); 272 fn(dl, dh, al, ah, bl, bh); 273 write_vec_element_i64(dh, d, 0, ES_64); 274 write_vec_element_i64(dl, d, 1, ES_64); 275 276 tcg_temp_free_i64(dh); 277 tcg_temp_free_i64(dl); 278 tcg_temp_free_i64(ah); 279 tcg_temp_free_i64(al); 280 tcg_temp_free_i64(bh); 281 tcg_temp_free_i64(bl); 282} 283 284typedef void (*gen_gvec128_4_i64_fn)(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, 285 TCGv_i64 ah, TCGv_i64 bl, TCGv_i64 bh, 286 TCGv_i64 cl, TCGv_i64 ch); 287static void gen_gvec128_4_i64(gen_gvec128_4_i64_fn fn, uint8_t d, uint8_t a, 288 uint8_t b, uint8_t c) 289{ 290 TCGv_i64 dh = tcg_temp_new_i64(); 291 TCGv_i64 dl = tcg_temp_new_i64(); 292 TCGv_i64 ah = tcg_temp_new_i64(); 293 TCGv_i64 al = tcg_temp_new_i64(); 294 TCGv_i64 bh = tcg_temp_new_i64(); 295 TCGv_i64 bl = tcg_temp_new_i64(); 296 TCGv_i64 ch = tcg_temp_new_i64(); 297 TCGv_i64 cl = tcg_temp_new_i64(); 298 299 read_vec_element_i64(ah, a, 0, ES_64); 300 read_vec_element_i64(al, a, 1, ES_64); 301 read_vec_element_i64(bh, b, 0, ES_64); 302 read_vec_element_i64(bl, b, 1, ES_64); 303 read_vec_element_i64(ch, c, 0, ES_64); 304 read_vec_element_i64(cl, c, 1, ES_64); 305 fn(dl, dh, al, ah, bl, bh, cl, ch); 306 write_vec_element_i64(dh, d, 0, ES_64); 307 write_vec_element_i64(dl, d, 1, ES_64); 308 309 tcg_temp_free_i64(dh); 310 tcg_temp_free_i64(dl); 311 tcg_temp_free_i64(ah); 312 tcg_temp_free_i64(al); 313 tcg_temp_free_i64(bh); 314 tcg_temp_free_i64(bl); 315 tcg_temp_free_i64(ch); 316 tcg_temp_free_i64(cl); 317} 318 319static void gen_addi2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, TCGv_i64 ah, 320 uint64_t b) 321{ 322 TCGv_i64 bl = tcg_const_i64(b); 323 TCGv_i64 bh = tcg_const_i64(0); 324 325 tcg_gen_add2_i64(dl, dh, al, ah, bl, bh); 326 tcg_temp_free_i64(bl); 327 tcg_temp_free_i64(bh); 328} 329 330static DisasJumpType op_vbperm(DisasContext *s, DisasOps *o) 331{ 332 gen_gvec_3_ool(get_field(s, v1), get_field(s, v2), get_field(s, v3), 0, 333 gen_helper_gvec_vbperm); 334 335 return DISAS_NEXT; 336} 337 338static DisasJumpType op_vge(DisasContext *s, DisasOps *o) 339{ 340 const uint8_t es = s->insn->data; 341 const uint8_t enr = get_field(s, m3); 342 TCGv_i64 tmp; 343 344 if (!valid_vec_element(enr, es)) { 345 gen_program_exception(s, PGM_SPECIFICATION); 346 return DISAS_NORETURN; 347 } 348 349 tmp = tcg_temp_new_i64(); 350 read_vec_element_i64(tmp, get_field(s, v2), enr, es); 351 tcg_gen_add_i64(o->addr1, o->addr1, tmp); 352 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 0); 353 354 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_TE | es); 355 write_vec_element_i64(tmp, get_field(s, v1), enr, es); 356 tcg_temp_free_i64(tmp); 357 return DISAS_NEXT; 358} 359 360static uint64_t generate_byte_mask(uint8_t mask) 361{ 362 uint64_t r = 0; 363 int i; 364 365 for (i = 0; i < 8; i++) { 366 if ((mask >> i) & 1) { 367 r |= 0xffull << (i * 8); 368 } 369 } 370 return r; 371} 372 373static DisasJumpType op_vgbm(DisasContext *s, DisasOps *o) 374{ 375 const uint16_t i2 = get_field(s, i2); 376 377 if (i2 == (i2 & 0xff) * 0x0101) { 378 /* 379 * Masks for both 64 bit elements of the vector are the same. 380 * Trust tcg to produce a good constant loading. 381 */ 382 gen_gvec_dup_imm(ES_64, get_field(s, v1), 383 generate_byte_mask(i2 & 0xff)); 384 } else { 385 TCGv_i64 t = tcg_temp_new_i64(); 386 387 tcg_gen_movi_i64(t, generate_byte_mask(i2 >> 8)); 388 write_vec_element_i64(t, get_field(s, v1), 0, ES_64); 389 tcg_gen_movi_i64(t, generate_byte_mask(i2)); 390 write_vec_element_i64(t, get_field(s, v1), 1, ES_64); 391 tcg_temp_free_i64(t); 392 } 393 return DISAS_NEXT; 394} 395 396static DisasJumpType op_vgm(DisasContext *s, DisasOps *o) 397{ 398 const uint8_t es = get_field(s, m4); 399 const uint8_t bits = NUM_VEC_ELEMENT_BITS(es); 400 const uint8_t i2 = get_field(s, i2) & (bits - 1); 401 const uint8_t i3 = get_field(s, i3) & (bits - 1); 402 uint64_t mask = 0; 403 int i; 404 405 if (es > ES_64) { 406 gen_program_exception(s, PGM_SPECIFICATION); 407 return DISAS_NORETURN; 408 } 409 410 /* generate the mask - take care of wrapping */ 411 for (i = i2; ; i = (i + 1) % bits) { 412 mask |= 1ull << (bits - i - 1); 413 if (i == i3) { 414 break; 415 } 416 } 417 418 gen_gvec_dup_imm(es, get_field(s, v1), mask); 419 return DISAS_NEXT; 420} 421 422static DisasJumpType op_vl(DisasContext *s, DisasOps *o) 423{ 424 TCGv_i64 t0 = tcg_temp_new_i64(); 425 TCGv_i64 t1 = tcg_temp_new_i64(); 426 427 tcg_gen_qemu_ld_i64(t0, o->addr1, get_mem_index(s), MO_TEUQ); 428 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 429 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_TEUQ); 430 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 431 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 432 tcg_temp_free(t0); 433 tcg_temp_free(t1); 434 return DISAS_NEXT; 435} 436 437static DisasJumpType op_vlr(DisasContext *s, DisasOps *o) 438{ 439 gen_gvec_mov(get_field(s, v1), get_field(s, v2)); 440 return DISAS_NEXT; 441} 442 443static DisasJumpType op_vlrep(DisasContext *s, DisasOps *o) 444{ 445 const uint8_t es = get_field(s, m3); 446 TCGv_i64 tmp; 447 448 if (es > ES_64) { 449 gen_program_exception(s, PGM_SPECIFICATION); 450 return DISAS_NORETURN; 451 } 452 453 tmp = tcg_temp_new_i64(); 454 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_TE | es); 455 gen_gvec_dup_i64(es, get_field(s, v1), tmp); 456 tcg_temp_free_i64(tmp); 457 return DISAS_NEXT; 458} 459 460static DisasJumpType op_vlebr(DisasContext *s, DisasOps *o) 461{ 462 const uint8_t es = s->insn->data; 463 const uint8_t enr = get_field(s, m3); 464 TCGv_i64 tmp; 465 466 if (!valid_vec_element(enr, es)) { 467 gen_program_exception(s, PGM_SPECIFICATION); 468 return DISAS_NORETURN; 469 } 470 471 tmp = tcg_temp_new_i64(); 472 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_LE | es); 473 write_vec_element_i64(tmp, get_field(s, v1), enr, es); 474 tcg_temp_free_i64(tmp); 475 return DISAS_NEXT; 476} 477 478static DisasJumpType op_vlbrrep(DisasContext *s, DisasOps *o) 479{ 480 const uint8_t es = get_field(s, m3); 481 TCGv_i64 tmp; 482 483 if (es < ES_16 || es > ES_64) { 484 gen_program_exception(s, PGM_SPECIFICATION); 485 return DISAS_NORETURN; 486 } 487 488 tmp = tcg_temp_new_i64(); 489 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_LE | es); 490 gen_gvec_dup_i64(es, get_field(s, v1), tmp); 491 tcg_temp_free_i64(tmp); 492 return DISAS_NEXT; 493} 494 495static DisasJumpType op_vllebrz(DisasContext *s, DisasOps *o) 496{ 497 const uint8_t m3 = get_field(s, m3); 498 TCGv_i64 tmp; 499 int es, lshift; 500 501 switch (m3) { 502 case ES_16: 503 case ES_32: 504 case ES_64: 505 es = m3; 506 lshift = 0; 507 break; 508 case 6: 509 es = ES_32; 510 lshift = 32; 511 break; 512 default: 513 gen_program_exception(s, PGM_SPECIFICATION); 514 return DISAS_NORETURN; 515 } 516 517 tmp = tcg_temp_new_i64(); 518 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_LE | es); 519 tcg_gen_shli_i64(tmp, tmp, lshift); 520 521 write_vec_element_i64(tmp, get_field(s, v1), 0, ES_64); 522 write_vec_element_i64(tcg_constant_i64(0), get_field(s, v1), 1, ES_64); 523 tcg_temp_free_i64(tmp); 524 return DISAS_NEXT; 525} 526 527static DisasJumpType op_vlbr(DisasContext *s, DisasOps *o) 528{ 529 const uint8_t es = get_field(s, m3); 530 TCGv_i64 t0, t1; 531 532 if (es < ES_16 || es > ES_128) { 533 gen_program_exception(s, PGM_SPECIFICATION); 534 return DISAS_NORETURN; 535 } 536 537 t0 = tcg_temp_new_i64(); 538 t1 = tcg_temp_new_i64(); 539 540 541 if (es == ES_128) { 542 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_LEUQ); 543 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 544 tcg_gen_qemu_ld_i64(t0, o->addr1, get_mem_index(s), MO_LEUQ); 545 goto write; 546 } 547 548 /* Begin with byte reversed doublewords... */ 549 tcg_gen_qemu_ld_i64(t0, o->addr1, get_mem_index(s), MO_LEUQ); 550 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 551 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_LEUQ); 552 553 /* 554 * For 16 and 32-bit elements, the doubleword bswap also reversed 555 * the order of the elements. Perform a larger order swap to put 556 * them back into place. For the 128-bit "element", finish the 557 * bswap by swapping the doublewords. 558 */ 559 switch (es) { 560 case ES_16: 561 tcg_gen_hswap_i64(t0, t0); 562 tcg_gen_hswap_i64(t1, t1); 563 break; 564 case ES_32: 565 tcg_gen_wswap_i64(t0, t0); 566 tcg_gen_wswap_i64(t1, t1); 567 break; 568 case ES_64: 569 break; 570 default: 571 g_assert_not_reached(); 572 } 573 574write: 575 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 576 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 577 578 tcg_temp_free(t0); 579 tcg_temp_free(t1); 580 return DISAS_NEXT; 581} 582 583static DisasJumpType op_vle(DisasContext *s, DisasOps *o) 584{ 585 const uint8_t es = s->insn->data; 586 const uint8_t enr = get_field(s, m3); 587 TCGv_i64 tmp; 588 589 if (!valid_vec_element(enr, es)) { 590 gen_program_exception(s, PGM_SPECIFICATION); 591 return DISAS_NORETURN; 592 } 593 594 tmp = tcg_temp_new_i64(); 595 tcg_gen_qemu_ld_i64(tmp, o->addr1, get_mem_index(s), MO_TE | es); 596 write_vec_element_i64(tmp, get_field(s, v1), enr, es); 597 tcg_temp_free_i64(tmp); 598 return DISAS_NEXT; 599} 600 601static DisasJumpType op_vlei(DisasContext *s, DisasOps *o) 602{ 603 const uint8_t es = s->insn->data; 604 const uint8_t enr = get_field(s, m3); 605 TCGv_i64 tmp; 606 607 if (!valid_vec_element(enr, es)) { 608 gen_program_exception(s, PGM_SPECIFICATION); 609 return DISAS_NORETURN; 610 } 611 612 tmp = tcg_const_i64((int16_t)get_field(s, i2)); 613 write_vec_element_i64(tmp, get_field(s, v1), enr, es); 614 tcg_temp_free_i64(tmp); 615 return DISAS_NEXT; 616} 617 618static DisasJumpType op_vler(DisasContext *s, DisasOps *o) 619{ 620 const uint8_t es = get_field(s, m3); 621 622 if (es < ES_16 || es > ES_64) { 623 gen_program_exception(s, PGM_SPECIFICATION); 624 return DISAS_NORETURN; 625 } 626 627 TCGv_i64 t0 = tcg_temp_new_i64(); 628 TCGv_i64 t1 = tcg_temp_new_i64(); 629 630 /* Begin with the two doublewords swapped... */ 631 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_TEUQ); 632 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 633 tcg_gen_qemu_ld_i64(t0, o->addr1, get_mem_index(s), MO_TEUQ); 634 635 /* ... then swap smaller elements within the doublewords as required. */ 636 switch (es) { 637 case MO_16: 638 tcg_gen_hswap_i64(t1, t1); 639 tcg_gen_hswap_i64(t0, t0); 640 break; 641 case MO_32: 642 tcg_gen_wswap_i64(t1, t1); 643 tcg_gen_wswap_i64(t0, t0); 644 break; 645 case MO_64: 646 break; 647 default: 648 g_assert_not_reached(); 649 } 650 651 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 652 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 653 tcg_temp_free(t0); 654 tcg_temp_free(t1); 655 return DISAS_NEXT; 656} 657 658static DisasJumpType op_vlgv(DisasContext *s, DisasOps *o) 659{ 660 const uint8_t es = get_field(s, m4); 661 TCGv_ptr ptr; 662 663 if (es > ES_64) { 664 gen_program_exception(s, PGM_SPECIFICATION); 665 return DISAS_NORETURN; 666 } 667 668 /* fast path if we don't need the register content */ 669 if (!get_field(s, b2)) { 670 uint8_t enr = get_field(s, d2) & (NUM_VEC_ELEMENTS(es) - 1); 671 672 read_vec_element_i64(o->out, get_field(s, v3), enr, es); 673 return DISAS_NEXT; 674 } 675 676 ptr = tcg_temp_new_ptr(); 677 get_vec_element_ptr_i64(ptr, get_field(s, v3), o->addr1, es); 678 switch (es) { 679 case ES_8: 680 tcg_gen_ld8u_i64(o->out, ptr, 0); 681 break; 682 case ES_16: 683 tcg_gen_ld16u_i64(o->out, ptr, 0); 684 break; 685 case ES_32: 686 tcg_gen_ld32u_i64(o->out, ptr, 0); 687 break; 688 case ES_64: 689 tcg_gen_ld_i64(o->out, ptr, 0); 690 break; 691 default: 692 g_assert_not_reached(); 693 } 694 tcg_temp_free_ptr(ptr); 695 696 return DISAS_NEXT; 697} 698 699static DisasJumpType op_vllez(DisasContext *s, DisasOps *o) 700{ 701 uint8_t es = get_field(s, m3); 702 uint8_t enr; 703 TCGv_i64 t; 704 705 switch (es) { 706 /* rightmost sub-element of leftmost doubleword */ 707 case ES_8: 708 enr = 7; 709 break; 710 case ES_16: 711 enr = 3; 712 break; 713 case ES_32: 714 enr = 1; 715 break; 716 case ES_64: 717 enr = 0; 718 break; 719 /* leftmost sub-element of leftmost doubleword */ 720 case 6: 721 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 722 es = ES_32; 723 enr = 0; 724 break; 725 } 726 /* fallthrough */ 727 default: 728 gen_program_exception(s, PGM_SPECIFICATION); 729 return DISAS_NORETURN; 730 } 731 732 t = tcg_temp_new_i64(); 733 tcg_gen_qemu_ld_i64(t, o->addr1, get_mem_index(s), MO_TE | es); 734 gen_gvec_dup_imm(es, get_field(s, v1), 0); 735 write_vec_element_i64(t, get_field(s, v1), enr, es); 736 tcg_temp_free_i64(t); 737 return DISAS_NEXT; 738} 739 740static DisasJumpType op_vlm(DisasContext *s, DisasOps *o) 741{ 742 const uint8_t v3 = get_field(s, v3); 743 uint8_t v1 = get_field(s, v1); 744 TCGv_i64 t0, t1; 745 746 if (v3 < v1 || (v3 - v1 + 1) > 16) { 747 gen_program_exception(s, PGM_SPECIFICATION); 748 return DISAS_NORETURN; 749 } 750 751 /* 752 * Check for possible access exceptions by trying to load the last 753 * element. The first element will be checked first next. 754 */ 755 t0 = tcg_temp_new_i64(); 756 t1 = tcg_temp_new_i64(); 757 gen_addi_and_wrap_i64(s, t0, o->addr1, (v3 - v1) * 16 + 8); 758 tcg_gen_qemu_ld_i64(t0, t0, get_mem_index(s), MO_TEUQ); 759 760 for (;; v1++) { 761 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_TEUQ); 762 write_vec_element_i64(t1, v1, 0, ES_64); 763 if (v1 == v3) { 764 break; 765 } 766 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 767 tcg_gen_qemu_ld_i64(t1, o->addr1, get_mem_index(s), MO_TEUQ); 768 write_vec_element_i64(t1, v1, 1, ES_64); 769 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 770 } 771 772 /* Store the last element, loaded first */ 773 write_vec_element_i64(t0, v1, 1, ES_64); 774 775 tcg_temp_free_i64(t0); 776 tcg_temp_free_i64(t1); 777 return DISAS_NEXT; 778} 779 780static DisasJumpType op_vlbb(DisasContext *s, DisasOps *o) 781{ 782 const int64_t block_size = (1ull << (get_field(s, m3) + 6)); 783 const int v1_offs = vec_full_reg_offset(get_field(s, v1)); 784 TCGv_ptr a0; 785 TCGv_i64 bytes; 786 787 if (get_field(s, m3) > 6) { 788 gen_program_exception(s, PGM_SPECIFICATION); 789 return DISAS_NORETURN; 790 } 791 792 bytes = tcg_temp_new_i64(); 793 a0 = tcg_temp_new_ptr(); 794 /* calculate the number of bytes until the next block boundary */ 795 tcg_gen_ori_i64(bytes, o->addr1, -block_size); 796 tcg_gen_neg_i64(bytes, bytes); 797 798 tcg_gen_addi_ptr(a0, cpu_env, v1_offs); 799 gen_helper_vll(cpu_env, a0, o->addr1, bytes); 800 tcg_temp_free_i64(bytes); 801 tcg_temp_free_ptr(a0); 802 return DISAS_NEXT; 803} 804 805static DisasJumpType op_vlvg(DisasContext *s, DisasOps *o) 806{ 807 const uint8_t es = get_field(s, m4); 808 TCGv_ptr ptr; 809 810 if (es > ES_64) { 811 gen_program_exception(s, PGM_SPECIFICATION); 812 return DISAS_NORETURN; 813 } 814 815 /* fast path if we don't need the register content */ 816 if (!get_field(s, b2)) { 817 uint8_t enr = get_field(s, d2) & (NUM_VEC_ELEMENTS(es) - 1); 818 819 write_vec_element_i64(o->in2, get_field(s, v1), enr, es); 820 return DISAS_NEXT; 821 } 822 823 ptr = tcg_temp_new_ptr(); 824 get_vec_element_ptr_i64(ptr, get_field(s, v1), o->addr1, es); 825 switch (es) { 826 case ES_8: 827 tcg_gen_st8_i64(o->in2, ptr, 0); 828 break; 829 case ES_16: 830 tcg_gen_st16_i64(o->in2, ptr, 0); 831 break; 832 case ES_32: 833 tcg_gen_st32_i64(o->in2, ptr, 0); 834 break; 835 case ES_64: 836 tcg_gen_st_i64(o->in2, ptr, 0); 837 break; 838 default: 839 g_assert_not_reached(); 840 } 841 tcg_temp_free_ptr(ptr); 842 843 return DISAS_NEXT; 844} 845 846static DisasJumpType op_vlvgp(DisasContext *s, DisasOps *o) 847{ 848 write_vec_element_i64(o->in1, get_field(s, v1), 0, ES_64); 849 write_vec_element_i64(o->in2, get_field(s, v1), 1, ES_64); 850 return DISAS_NEXT; 851} 852 853static DisasJumpType op_vll(DisasContext *s, DisasOps *o) 854{ 855 const int v1_offs = vec_full_reg_offset(get_field(s, v1)); 856 TCGv_ptr a0 = tcg_temp_new_ptr(); 857 858 /* convert highest index into an actual length */ 859 tcg_gen_addi_i64(o->in2, o->in2, 1); 860 tcg_gen_addi_ptr(a0, cpu_env, v1_offs); 861 gen_helper_vll(cpu_env, a0, o->addr1, o->in2); 862 tcg_temp_free_ptr(a0); 863 return DISAS_NEXT; 864} 865 866static DisasJumpType op_vmr(DisasContext *s, DisasOps *o) 867{ 868 const uint8_t v1 = get_field(s, v1); 869 const uint8_t v2 = get_field(s, v2); 870 const uint8_t v3 = get_field(s, v3); 871 const uint8_t es = get_field(s, m4); 872 int dst_idx, src_idx; 873 TCGv_i64 tmp; 874 875 if (es > ES_64) { 876 gen_program_exception(s, PGM_SPECIFICATION); 877 return DISAS_NORETURN; 878 } 879 880 tmp = tcg_temp_new_i64(); 881 if (s->fields.op2 == 0x61) { 882 /* iterate backwards to avoid overwriting data we might need later */ 883 for (dst_idx = NUM_VEC_ELEMENTS(es) - 1; dst_idx >= 0; dst_idx--) { 884 src_idx = dst_idx / 2; 885 if (dst_idx % 2 == 0) { 886 read_vec_element_i64(tmp, v2, src_idx, es); 887 } else { 888 read_vec_element_i64(tmp, v3, src_idx, es); 889 } 890 write_vec_element_i64(tmp, v1, dst_idx, es); 891 } 892 } else { 893 /* iterate forward to avoid overwriting data we might need later */ 894 for (dst_idx = 0; dst_idx < NUM_VEC_ELEMENTS(es); dst_idx++) { 895 src_idx = (dst_idx + NUM_VEC_ELEMENTS(es)) / 2; 896 if (dst_idx % 2 == 0) { 897 read_vec_element_i64(tmp, v2, src_idx, es); 898 } else { 899 read_vec_element_i64(tmp, v3, src_idx, es); 900 } 901 write_vec_element_i64(tmp, v1, dst_idx, es); 902 } 903 } 904 tcg_temp_free_i64(tmp); 905 return DISAS_NEXT; 906} 907 908static DisasJumpType op_vpk(DisasContext *s, DisasOps *o) 909{ 910 const uint8_t v1 = get_field(s, v1); 911 const uint8_t v2 = get_field(s, v2); 912 const uint8_t v3 = get_field(s, v3); 913 const uint8_t es = get_field(s, m4); 914 static gen_helper_gvec_3 * const vpk[3] = { 915 gen_helper_gvec_vpk16, 916 gen_helper_gvec_vpk32, 917 gen_helper_gvec_vpk64, 918 }; 919 static gen_helper_gvec_3 * const vpks[3] = { 920 gen_helper_gvec_vpks16, 921 gen_helper_gvec_vpks32, 922 gen_helper_gvec_vpks64, 923 }; 924 static gen_helper_gvec_3_ptr * const vpks_cc[3] = { 925 gen_helper_gvec_vpks_cc16, 926 gen_helper_gvec_vpks_cc32, 927 gen_helper_gvec_vpks_cc64, 928 }; 929 static gen_helper_gvec_3 * const vpkls[3] = { 930 gen_helper_gvec_vpkls16, 931 gen_helper_gvec_vpkls32, 932 gen_helper_gvec_vpkls64, 933 }; 934 static gen_helper_gvec_3_ptr * const vpkls_cc[3] = { 935 gen_helper_gvec_vpkls_cc16, 936 gen_helper_gvec_vpkls_cc32, 937 gen_helper_gvec_vpkls_cc64, 938 }; 939 940 if (es == ES_8 || es > ES_64) { 941 gen_program_exception(s, PGM_SPECIFICATION); 942 return DISAS_NORETURN; 943 } 944 945 switch (s->fields.op2) { 946 case 0x97: 947 if (get_field(s, m5) & 0x1) { 948 gen_gvec_3_ptr(v1, v2, v3, cpu_env, 0, vpks_cc[es - 1]); 949 set_cc_static(s); 950 } else { 951 gen_gvec_3_ool(v1, v2, v3, 0, vpks[es - 1]); 952 } 953 break; 954 case 0x95: 955 if (get_field(s, m5) & 0x1) { 956 gen_gvec_3_ptr(v1, v2, v3, cpu_env, 0, vpkls_cc[es - 1]); 957 set_cc_static(s); 958 } else { 959 gen_gvec_3_ool(v1, v2, v3, 0, vpkls[es - 1]); 960 } 961 break; 962 case 0x94: 963 /* If sources and destination dont't overlap -> fast path */ 964 if (v1 != v2 && v1 != v3) { 965 const uint8_t src_es = get_field(s, m4); 966 const uint8_t dst_es = src_es - 1; 967 TCGv_i64 tmp = tcg_temp_new_i64(); 968 int dst_idx, src_idx; 969 970 for (dst_idx = 0; dst_idx < NUM_VEC_ELEMENTS(dst_es); dst_idx++) { 971 src_idx = dst_idx; 972 if (src_idx < NUM_VEC_ELEMENTS(src_es)) { 973 read_vec_element_i64(tmp, v2, src_idx, src_es); 974 } else { 975 src_idx -= NUM_VEC_ELEMENTS(src_es); 976 read_vec_element_i64(tmp, v3, src_idx, src_es); 977 } 978 write_vec_element_i64(tmp, v1, dst_idx, dst_es); 979 } 980 tcg_temp_free_i64(tmp); 981 } else { 982 gen_gvec_3_ool(v1, v2, v3, 0, vpk[es - 1]); 983 } 984 break; 985 default: 986 g_assert_not_reached(); 987 } 988 return DISAS_NEXT; 989} 990 991static DisasJumpType op_vperm(DisasContext *s, DisasOps *o) 992{ 993 gen_gvec_4_ool(get_field(s, v1), get_field(s, v2), 994 get_field(s, v3), get_field(s, v4), 995 0, gen_helper_gvec_vperm); 996 return DISAS_NEXT; 997} 998 999static DisasJumpType op_vpdi(DisasContext *s, DisasOps *o) 1000{ 1001 const uint8_t i2 = extract32(get_field(s, m4), 2, 1); 1002 const uint8_t i3 = extract32(get_field(s, m4), 0, 1); 1003 TCGv_i64 t0 = tcg_temp_new_i64(); 1004 TCGv_i64 t1 = tcg_temp_new_i64(); 1005 1006 read_vec_element_i64(t0, get_field(s, v2), i2, ES_64); 1007 read_vec_element_i64(t1, get_field(s, v3), i3, ES_64); 1008 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 1009 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 1010 tcg_temp_free_i64(t0); 1011 tcg_temp_free_i64(t1); 1012 return DISAS_NEXT; 1013} 1014 1015static DisasJumpType op_vrep(DisasContext *s, DisasOps *o) 1016{ 1017 const uint8_t enr = get_field(s, i2); 1018 const uint8_t es = get_field(s, m4); 1019 1020 if (es > ES_64 || !valid_vec_element(enr, es)) { 1021 gen_program_exception(s, PGM_SPECIFICATION); 1022 return DISAS_NORETURN; 1023 } 1024 1025 tcg_gen_gvec_dup_mem(es, vec_full_reg_offset(get_field(s, v1)), 1026 vec_reg_offset(get_field(s, v3), enr, es), 1027 16, 16); 1028 return DISAS_NEXT; 1029} 1030 1031static DisasJumpType op_vrepi(DisasContext *s, DisasOps *o) 1032{ 1033 const int64_t data = (int16_t)get_field(s, i2); 1034 const uint8_t es = get_field(s, m3); 1035 1036 if (es > ES_64) { 1037 gen_program_exception(s, PGM_SPECIFICATION); 1038 return DISAS_NORETURN; 1039 } 1040 1041 gen_gvec_dup_imm(es, get_field(s, v1), data); 1042 return DISAS_NEXT; 1043} 1044 1045static DisasJumpType op_vsce(DisasContext *s, DisasOps *o) 1046{ 1047 const uint8_t es = s->insn->data; 1048 const uint8_t enr = get_field(s, m3); 1049 TCGv_i64 tmp; 1050 1051 if (!valid_vec_element(enr, es)) { 1052 gen_program_exception(s, PGM_SPECIFICATION); 1053 return DISAS_NORETURN; 1054 } 1055 1056 tmp = tcg_temp_new_i64(); 1057 read_vec_element_i64(tmp, get_field(s, v2), enr, es); 1058 tcg_gen_add_i64(o->addr1, o->addr1, tmp); 1059 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 0); 1060 1061 read_vec_element_i64(tmp, get_field(s, v1), enr, es); 1062 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TE | es); 1063 tcg_temp_free_i64(tmp); 1064 return DISAS_NEXT; 1065} 1066 1067static DisasJumpType op_vsel(DisasContext *s, DisasOps *o) 1068{ 1069 gen_gvec_fn_4(bitsel, ES_8, get_field(s, v1), 1070 get_field(s, v4), get_field(s, v2), 1071 get_field(s, v3)); 1072 return DISAS_NEXT; 1073} 1074 1075static DisasJumpType op_vseg(DisasContext *s, DisasOps *o) 1076{ 1077 const uint8_t es = get_field(s, m3); 1078 int idx1, idx2; 1079 TCGv_i64 tmp; 1080 1081 switch (es) { 1082 case ES_8: 1083 idx1 = 7; 1084 idx2 = 15; 1085 break; 1086 case ES_16: 1087 idx1 = 3; 1088 idx2 = 7; 1089 break; 1090 case ES_32: 1091 idx1 = 1; 1092 idx2 = 3; 1093 break; 1094 default: 1095 gen_program_exception(s, PGM_SPECIFICATION); 1096 return DISAS_NORETURN; 1097 } 1098 1099 tmp = tcg_temp_new_i64(); 1100 read_vec_element_i64(tmp, get_field(s, v2), idx1, es | MO_SIGN); 1101 write_vec_element_i64(tmp, get_field(s, v1), 0, ES_64); 1102 read_vec_element_i64(tmp, get_field(s, v2), idx2, es | MO_SIGN); 1103 write_vec_element_i64(tmp, get_field(s, v1), 1, ES_64); 1104 tcg_temp_free_i64(tmp); 1105 return DISAS_NEXT; 1106} 1107 1108static DisasJumpType op_vst(DisasContext *s, DisasOps *o) 1109{ 1110 TCGv_i64 tmp = tcg_const_i64(16); 1111 1112 /* Probe write access before actually modifying memory */ 1113 gen_helper_probe_write_access(cpu_env, o->addr1, tmp); 1114 1115 read_vec_element_i64(tmp, get_field(s, v1), 0, ES_64); 1116 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TEUQ); 1117 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 1118 read_vec_element_i64(tmp, get_field(s, v1), 1, ES_64); 1119 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TEUQ); 1120 tcg_temp_free_i64(tmp); 1121 return DISAS_NEXT; 1122} 1123 1124static DisasJumpType op_vstebr(DisasContext *s, DisasOps *o) 1125{ 1126 const uint8_t es = s->insn->data; 1127 const uint8_t enr = get_field(s, m3); 1128 TCGv_i64 tmp; 1129 1130 if (!valid_vec_element(enr, es)) { 1131 gen_program_exception(s, PGM_SPECIFICATION); 1132 return DISAS_NORETURN; 1133 } 1134 1135 tmp = tcg_temp_new_i64(); 1136 read_vec_element_i64(tmp, get_field(s, v1), enr, es); 1137 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_LE | es); 1138 tcg_temp_free_i64(tmp); 1139 return DISAS_NEXT; 1140} 1141 1142static DisasJumpType op_vstbr(DisasContext *s, DisasOps *o) 1143{ 1144 const uint8_t es = get_field(s, m3); 1145 TCGv_i64 t0, t1; 1146 1147 if (es < ES_16 || es > ES_128) { 1148 gen_program_exception(s, PGM_SPECIFICATION); 1149 return DISAS_NORETURN; 1150 } 1151 1152 /* Probe write access before actually modifying memory */ 1153 gen_helper_probe_write_access(cpu_env, o->addr1, tcg_constant_i64(16)); 1154 1155 t0 = tcg_temp_new_i64(); 1156 t1 = tcg_temp_new_i64(); 1157 1158 1159 if (es == ES_128) { 1160 read_vec_element_i64(t1, get_field(s, v1), 0, ES_64); 1161 read_vec_element_i64(t0, get_field(s, v1), 1, ES_64); 1162 goto write; 1163 } 1164 1165 read_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 1166 read_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 1167 1168 /* 1169 * For 16 and 32-bit elements, the doubleword bswap below will 1170 * reverse the order of the elements. Perform a larger order 1171 * swap to put them back into place. For the 128-bit "element", 1172 * finish the bswap by swapping the doublewords. 1173 */ 1174 switch (es) { 1175 case MO_16: 1176 tcg_gen_hswap_i64(t0, t0); 1177 tcg_gen_hswap_i64(t1, t1); 1178 break; 1179 case MO_32: 1180 tcg_gen_wswap_i64(t0, t0); 1181 tcg_gen_wswap_i64(t1, t1); 1182 break; 1183 case MO_64: 1184 break; 1185 default: 1186 g_assert_not_reached(); 1187 } 1188 1189write: 1190 tcg_gen_qemu_st_i64(t0, o->addr1, get_mem_index(s), MO_LEUQ); 1191 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 1192 tcg_gen_qemu_st_i64(t1, o->addr1, get_mem_index(s), MO_LEUQ); 1193 1194 tcg_temp_free(t0); 1195 tcg_temp_free(t1); 1196 return DISAS_NEXT; 1197} 1198 1199static DisasJumpType op_vste(DisasContext *s, DisasOps *o) 1200{ 1201 const uint8_t es = s->insn->data; 1202 const uint8_t enr = get_field(s, m3); 1203 TCGv_i64 tmp; 1204 1205 if (!valid_vec_element(enr, es)) { 1206 gen_program_exception(s, PGM_SPECIFICATION); 1207 return DISAS_NORETURN; 1208 } 1209 1210 tmp = tcg_temp_new_i64(); 1211 read_vec_element_i64(tmp, get_field(s, v1), enr, es); 1212 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TE | es); 1213 tcg_temp_free_i64(tmp); 1214 return DISAS_NEXT; 1215} 1216 1217static DisasJumpType op_vster(DisasContext *s, DisasOps *o) 1218{ 1219 const uint8_t es = get_field(s, m3); 1220 TCGv_i64 t0, t1; 1221 1222 if (es < ES_16 || es > ES_64) { 1223 gen_program_exception(s, PGM_SPECIFICATION); 1224 return DISAS_NORETURN; 1225 } 1226 1227 /* Probe write access before actually modifying memory */ 1228 gen_helper_probe_write_access(cpu_env, o->addr1, tcg_constant_i64(16)); 1229 1230 /* Begin with the two doublewords swapped... */ 1231 t0 = tcg_temp_new_i64(); 1232 t1 = tcg_temp_new_i64(); 1233 read_vec_element_i64(t1, get_field(s, v1), 0, ES_64); 1234 read_vec_element_i64(t0, get_field(s, v1), 1, ES_64); 1235 1236 /* ... then swap smaller elements within the doublewords as required. */ 1237 switch (es) { 1238 case MO_16: 1239 tcg_gen_hswap_i64(t1, t1); 1240 tcg_gen_hswap_i64(t0, t0); 1241 break; 1242 case MO_32: 1243 tcg_gen_wswap_i64(t1, t1); 1244 tcg_gen_wswap_i64(t0, t0); 1245 break; 1246 case MO_64: 1247 break; 1248 default: 1249 g_assert_not_reached(); 1250 } 1251 1252 tcg_gen_qemu_st_i64(t0, o->addr1, get_mem_index(s), MO_TEUQ); 1253 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 1254 tcg_gen_qemu_st_i64(t1, o->addr1, get_mem_index(s), MO_TEUQ); 1255 1256 tcg_temp_free(t0); 1257 tcg_temp_free(t1); 1258 return DISAS_NEXT; 1259} 1260 1261static DisasJumpType op_vstm(DisasContext *s, DisasOps *o) 1262{ 1263 const uint8_t v3 = get_field(s, v3); 1264 uint8_t v1 = get_field(s, v1); 1265 TCGv_i64 tmp; 1266 1267 while (v3 < v1 || (v3 - v1 + 1) > 16) { 1268 gen_program_exception(s, PGM_SPECIFICATION); 1269 return DISAS_NORETURN; 1270 } 1271 1272 /* Probe write access before actually modifying memory */ 1273 tmp = tcg_const_i64((v3 - v1 + 1) * 16); 1274 gen_helper_probe_write_access(cpu_env, o->addr1, tmp); 1275 1276 for (;; v1++) { 1277 read_vec_element_i64(tmp, v1, 0, ES_64); 1278 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TEUQ); 1279 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 1280 read_vec_element_i64(tmp, v1, 1, ES_64); 1281 tcg_gen_qemu_st_i64(tmp, o->addr1, get_mem_index(s), MO_TEUQ); 1282 if (v1 == v3) { 1283 break; 1284 } 1285 gen_addi_and_wrap_i64(s, o->addr1, o->addr1, 8); 1286 } 1287 tcg_temp_free_i64(tmp); 1288 return DISAS_NEXT; 1289} 1290 1291static DisasJumpType op_vstl(DisasContext *s, DisasOps *o) 1292{ 1293 const int v1_offs = vec_full_reg_offset(get_field(s, v1)); 1294 TCGv_ptr a0 = tcg_temp_new_ptr(); 1295 1296 /* convert highest index into an actual length */ 1297 tcg_gen_addi_i64(o->in2, o->in2, 1); 1298 tcg_gen_addi_ptr(a0, cpu_env, v1_offs); 1299 gen_helper_vstl(cpu_env, a0, o->addr1, o->in2); 1300 tcg_temp_free_ptr(a0); 1301 return DISAS_NEXT; 1302} 1303 1304static DisasJumpType op_vup(DisasContext *s, DisasOps *o) 1305{ 1306 const bool logical = s->fields.op2 == 0xd4 || s->fields.op2 == 0xd5; 1307 const uint8_t v1 = get_field(s, v1); 1308 const uint8_t v2 = get_field(s, v2); 1309 const uint8_t src_es = get_field(s, m3); 1310 const uint8_t dst_es = src_es + 1; 1311 int dst_idx, src_idx; 1312 TCGv_i64 tmp; 1313 1314 if (src_es > ES_32) { 1315 gen_program_exception(s, PGM_SPECIFICATION); 1316 return DISAS_NORETURN; 1317 } 1318 1319 tmp = tcg_temp_new_i64(); 1320 if (s->fields.op2 == 0xd7 || s->fields.op2 == 0xd5) { 1321 /* iterate backwards to avoid overwriting data we might need later */ 1322 for (dst_idx = NUM_VEC_ELEMENTS(dst_es) - 1; dst_idx >= 0; dst_idx--) { 1323 src_idx = dst_idx; 1324 read_vec_element_i64(tmp, v2, src_idx, 1325 src_es | (logical ? 0 : MO_SIGN)); 1326 write_vec_element_i64(tmp, v1, dst_idx, dst_es); 1327 } 1328 1329 } else { 1330 /* iterate forward to avoid overwriting data we might need later */ 1331 for (dst_idx = 0; dst_idx < NUM_VEC_ELEMENTS(dst_es); dst_idx++) { 1332 src_idx = dst_idx + NUM_VEC_ELEMENTS(src_es) / 2; 1333 read_vec_element_i64(tmp, v2, src_idx, 1334 src_es | (logical ? 0 : MO_SIGN)); 1335 write_vec_element_i64(tmp, v1, dst_idx, dst_es); 1336 } 1337 } 1338 tcg_temp_free_i64(tmp); 1339 return DISAS_NEXT; 1340} 1341 1342static DisasJumpType op_va(DisasContext *s, DisasOps *o) 1343{ 1344 const uint8_t es = get_field(s, m4); 1345 1346 if (es > ES_128) { 1347 gen_program_exception(s, PGM_SPECIFICATION); 1348 return DISAS_NORETURN; 1349 } else if (es == ES_128) { 1350 gen_gvec128_3_i64(tcg_gen_add2_i64, get_field(s, v1), 1351 get_field(s, v2), get_field(s, v3)); 1352 return DISAS_NEXT; 1353 } 1354 gen_gvec_fn_3(add, es, get_field(s, v1), get_field(s, v2), 1355 get_field(s, v3)); 1356 return DISAS_NEXT; 1357} 1358 1359static void gen_acc(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b, uint8_t es) 1360{ 1361 const uint8_t msb_bit_nr = NUM_VEC_ELEMENT_BITS(es) - 1; 1362 TCGv_i64 msb_mask = tcg_const_i64(dup_const(es, 1ull << msb_bit_nr)); 1363 TCGv_i64 t1 = tcg_temp_new_i64(); 1364 TCGv_i64 t2 = tcg_temp_new_i64(); 1365 TCGv_i64 t3 = tcg_temp_new_i64(); 1366 1367 /* Calculate the carry into the MSB, ignoring the old MSBs */ 1368 tcg_gen_andc_i64(t1, a, msb_mask); 1369 tcg_gen_andc_i64(t2, b, msb_mask); 1370 tcg_gen_add_i64(t1, t1, t2); 1371 /* Calculate the MSB without any carry into it */ 1372 tcg_gen_xor_i64(t3, a, b); 1373 /* Calculate the carry out of the MSB in the MSB bit position */ 1374 tcg_gen_and_i64(d, a, b); 1375 tcg_gen_and_i64(t1, t1, t3); 1376 tcg_gen_or_i64(d, d, t1); 1377 /* Isolate and shift the carry into position */ 1378 tcg_gen_and_i64(d, d, msb_mask); 1379 tcg_gen_shri_i64(d, d, msb_bit_nr); 1380 1381 tcg_temp_free_i64(t1); 1382 tcg_temp_free_i64(t2); 1383 tcg_temp_free_i64(t3); 1384} 1385 1386static void gen_acc8_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b) 1387{ 1388 gen_acc(d, a, b, ES_8); 1389} 1390 1391static void gen_acc16_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b) 1392{ 1393 gen_acc(d, a, b, ES_16); 1394} 1395 1396static void gen_acc_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 1397{ 1398 TCGv_i32 t = tcg_temp_new_i32(); 1399 1400 tcg_gen_add_i32(t, a, b); 1401 tcg_gen_setcond_i32(TCG_COND_LTU, d, t, b); 1402 tcg_temp_free_i32(t); 1403} 1404 1405static void gen_acc_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b) 1406{ 1407 TCGv_i64 t = tcg_temp_new_i64(); 1408 1409 tcg_gen_add_i64(t, a, b); 1410 tcg_gen_setcond_i64(TCG_COND_LTU, d, t, b); 1411 tcg_temp_free_i64(t); 1412} 1413 1414static void gen_acc2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, 1415 TCGv_i64 ah, TCGv_i64 bl, TCGv_i64 bh) 1416{ 1417 TCGv_i64 th = tcg_temp_new_i64(); 1418 TCGv_i64 tl = tcg_temp_new_i64(); 1419 TCGv_i64 zero = tcg_const_i64(0); 1420 1421 tcg_gen_add2_i64(tl, th, al, zero, bl, zero); 1422 tcg_gen_add2_i64(tl, th, th, zero, ah, zero); 1423 tcg_gen_add2_i64(tl, dl, tl, th, bh, zero); 1424 tcg_gen_mov_i64(dh, zero); 1425 1426 tcg_temp_free_i64(th); 1427 tcg_temp_free_i64(tl); 1428 tcg_temp_free_i64(zero); 1429} 1430 1431static DisasJumpType op_vacc(DisasContext *s, DisasOps *o) 1432{ 1433 const uint8_t es = get_field(s, m4); 1434 static const GVecGen3 g[4] = { 1435 { .fni8 = gen_acc8_i64, }, 1436 { .fni8 = gen_acc16_i64, }, 1437 { .fni4 = gen_acc_i32, }, 1438 { .fni8 = gen_acc_i64, }, 1439 }; 1440 1441 if (es > ES_128) { 1442 gen_program_exception(s, PGM_SPECIFICATION); 1443 return DISAS_NORETURN; 1444 } else if (es == ES_128) { 1445 gen_gvec128_3_i64(gen_acc2_i64, get_field(s, v1), 1446 get_field(s, v2), get_field(s, v3)); 1447 return DISAS_NEXT; 1448 } 1449 gen_gvec_3(get_field(s, v1), get_field(s, v2), 1450 get_field(s, v3), &g[es]); 1451 return DISAS_NEXT; 1452} 1453 1454static void gen_ac2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, TCGv_i64 ah, 1455 TCGv_i64 bl, TCGv_i64 bh, TCGv_i64 cl, TCGv_i64 ch) 1456{ 1457 TCGv_i64 tl = tcg_temp_new_i64(); 1458 TCGv_i64 th = tcg_const_i64(0); 1459 1460 /* extract the carry only */ 1461 tcg_gen_extract_i64(tl, cl, 0, 1); 1462 tcg_gen_add2_i64(dl, dh, al, ah, bl, bh); 1463 tcg_gen_add2_i64(dl, dh, dl, dh, tl, th); 1464 1465 tcg_temp_free_i64(tl); 1466 tcg_temp_free_i64(th); 1467} 1468 1469static DisasJumpType op_vac(DisasContext *s, DisasOps *o) 1470{ 1471 if (get_field(s, m5) != ES_128) { 1472 gen_program_exception(s, PGM_SPECIFICATION); 1473 return DISAS_NORETURN; 1474 } 1475 1476 gen_gvec128_4_i64(gen_ac2_i64, get_field(s, v1), 1477 get_field(s, v2), get_field(s, v3), 1478 get_field(s, v4)); 1479 return DISAS_NEXT; 1480} 1481 1482static void gen_accc2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, TCGv_i64 ah, 1483 TCGv_i64 bl, TCGv_i64 bh, TCGv_i64 cl, TCGv_i64 ch) 1484{ 1485 TCGv_i64 tl = tcg_temp_new_i64(); 1486 TCGv_i64 th = tcg_temp_new_i64(); 1487 TCGv_i64 zero = tcg_const_i64(0); 1488 1489 tcg_gen_andi_i64(tl, cl, 1); 1490 tcg_gen_add2_i64(tl, th, tl, zero, al, zero); 1491 tcg_gen_add2_i64(tl, th, tl, th, bl, zero); 1492 tcg_gen_add2_i64(tl, th, th, zero, ah, zero); 1493 tcg_gen_add2_i64(tl, dl, tl, th, bh, zero); 1494 tcg_gen_mov_i64(dh, zero); 1495 1496 tcg_temp_free_i64(tl); 1497 tcg_temp_free_i64(th); 1498 tcg_temp_free_i64(zero); 1499} 1500 1501static DisasJumpType op_vaccc(DisasContext *s, DisasOps *o) 1502{ 1503 if (get_field(s, m5) != ES_128) { 1504 gen_program_exception(s, PGM_SPECIFICATION); 1505 return DISAS_NORETURN; 1506 } 1507 1508 gen_gvec128_4_i64(gen_accc2_i64, get_field(s, v1), 1509 get_field(s, v2), get_field(s, v3), 1510 get_field(s, v4)); 1511 return DISAS_NEXT; 1512} 1513 1514static DisasJumpType op_vn(DisasContext *s, DisasOps *o) 1515{ 1516 gen_gvec_fn_3(and, ES_8, get_field(s, v1), get_field(s, v2), 1517 get_field(s, v3)); 1518 return DISAS_NEXT; 1519} 1520 1521static DisasJumpType op_vnc(DisasContext *s, DisasOps *o) 1522{ 1523 gen_gvec_fn_3(andc, ES_8, get_field(s, v1), 1524 get_field(s, v2), get_field(s, v3)); 1525 return DISAS_NEXT; 1526} 1527 1528static void gen_avg_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 1529{ 1530 TCGv_i64 t0 = tcg_temp_new_i64(); 1531 TCGv_i64 t1 = tcg_temp_new_i64(); 1532 1533 tcg_gen_ext_i32_i64(t0, a); 1534 tcg_gen_ext_i32_i64(t1, b); 1535 tcg_gen_add_i64(t0, t0, t1); 1536 tcg_gen_addi_i64(t0, t0, 1); 1537 tcg_gen_shri_i64(t0, t0, 1); 1538 tcg_gen_extrl_i64_i32(d, t0); 1539 1540 tcg_temp_free(t0); 1541 tcg_temp_free(t1); 1542} 1543 1544static void gen_avg_i64(TCGv_i64 dl, TCGv_i64 al, TCGv_i64 bl) 1545{ 1546 TCGv_i64 dh = tcg_temp_new_i64(); 1547 TCGv_i64 ah = tcg_temp_new_i64(); 1548 TCGv_i64 bh = tcg_temp_new_i64(); 1549 1550 /* extending the sign by one bit is sufficient */ 1551 tcg_gen_extract_i64(ah, al, 63, 1); 1552 tcg_gen_extract_i64(bh, bl, 63, 1); 1553 tcg_gen_add2_i64(dl, dh, al, ah, bl, bh); 1554 gen_addi2_i64(dl, dh, dl, dh, 1); 1555 tcg_gen_extract2_i64(dl, dl, dh, 1); 1556 1557 tcg_temp_free_i64(dh); 1558 tcg_temp_free_i64(ah); 1559 tcg_temp_free_i64(bh); 1560} 1561 1562static DisasJumpType op_vavg(DisasContext *s, DisasOps *o) 1563{ 1564 const uint8_t es = get_field(s, m4); 1565 static const GVecGen3 g[4] = { 1566 { .fno = gen_helper_gvec_vavg8, }, 1567 { .fno = gen_helper_gvec_vavg16, }, 1568 { .fni4 = gen_avg_i32, }, 1569 { .fni8 = gen_avg_i64, }, 1570 }; 1571 1572 if (es > ES_64) { 1573 gen_program_exception(s, PGM_SPECIFICATION); 1574 return DISAS_NORETURN; 1575 } 1576 gen_gvec_3(get_field(s, v1), get_field(s, v2), 1577 get_field(s, v3), &g[es]); 1578 return DISAS_NEXT; 1579} 1580 1581static void gen_avgl_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 1582{ 1583 TCGv_i64 t0 = tcg_temp_new_i64(); 1584 TCGv_i64 t1 = tcg_temp_new_i64(); 1585 1586 tcg_gen_extu_i32_i64(t0, a); 1587 tcg_gen_extu_i32_i64(t1, b); 1588 tcg_gen_add_i64(t0, t0, t1); 1589 tcg_gen_addi_i64(t0, t0, 1); 1590 tcg_gen_shri_i64(t0, t0, 1); 1591 tcg_gen_extrl_i64_i32(d, t0); 1592 1593 tcg_temp_free(t0); 1594 tcg_temp_free(t1); 1595} 1596 1597static void gen_avgl_i64(TCGv_i64 dl, TCGv_i64 al, TCGv_i64 bl) 1598{ 1599 TCGv_i64 dh = tcg_temp_new_i64(); 1600 TCGv_i64 zero = tcg_const_i64(0); 1601 1602 tcg_gen_add2_i64(dl, dh, al, zero, bl, zero); 1603 gen_addi2_i64(dl, dh, dl, dh, 1); 1604 tcg_gen_extract2_i64(dl, dl, dh, 1); 1605 1606 tcg_temp_free_i64(dh); 1607 tcg_temp_free_i64(zero); 1608} 1609 1610static DisasJumpType op_vavgl(DisasContext *s, DisasOps *o) 1611{ 1612 const uint8_t es = get_field(s, m4); 1613 static const GVecGen3 g[4] = { 1614 { .fno = gen_helper_gvec_vavgl8, }, 1615 { .fno = gen_helper_gvec_vavgl16, }, 1616 { .fni4 = gen_avgl_i32, }, 1617 { .fni8 = gen_avgl_i64, }, 1618 }; 1619 1620 if (es > ES_64) { 1621 gen_program_exception(s, PGM_SPECIFICATION); 1622 return DISAS_NORETURN; 1623 } 1624 gen_gvec_3(get_field(s, v1), get_field(s, v2), 1625 get_field(s, v3), &g[es]); 1626 return DISAS_NEXT; 1627} 1628 1629static DisasJumpType op_vcksm(DisasContext *s, DisasOps *o) 1630{ 1631 TCGv_i32 tmp = tcg_temp_new_i32(); 1632 TCGv_i32 sum = tcg_temp_new_i32(); 1633 int i; 1634 1635 read_vec_element_i32(sum, get_field(s, v3), 1, ES_32); 1636 for (i = 0; i < 4; i++) { 1637 read_vec_element_i32(tmp, get_field(s, v2), i, ES_32); 1638 tcg_gen_add2_i32(tmp, sum, sum, sum, tmp, tmp); 1639 } 1640 gen_gvec_dup_imm(ES_32, get_field(s, v1), 0); 1641 write_vec_element_i32(sum, get_field(s, v1), 1, ES_32); 1642 1643 tcg_temp_free_i32(tmp); 1644 tcg_temp_free_i32(sum); 1645 return DISAS_NEXT; 1646} 1647 1648static DisasJumpType op_vec(DisasContext *s, DisasOps *o) 1649{ 1650 uint8_t es = get_field(s, m3); 1651 const uint8_t enr = NUM_VEC_ELEMENTS(es) / 2 - 1; 1652 1653 if (es > ES_64) { 1654 gen_program_exception(s, PGM_SPECIFICATION); 1655 return DISAS_NORETURN; 1656 } 1657 if (s->fields.op2 == 0xdb) { 1658 es |= MO_SIGN; 1659 } 1660 1661 o->in1 = tcg_temp_new_i64(); 1662 o->in2 = tcg_temp_new_i64(); 1663 read_vec_element_i64(o->in1, get_field(s, v1), enr, es); 1664 read_vec_element_i64(o->in2, get_field(s, v2), enr, es); 1665 return DISAS_NEXT; 1666} 1667 1668static DisasJumpType op_vc(DisasContext *s, DisasOps *o) 1669{ 1670 const uint8_t es = get_field(s, m4); 1671 TCGCond cond = s->insn->data; 1672 1673 if (es > ES_64) { 1674 gen_program_exception(s, PGM_SPECIFICATION); 1675 return DISAS_NORETURN; 1676 } 1677 1678 tcg_gen_gvec_cmp(cond, es, 1679 vec_full_reg_offset(get_field(s, v1)), 1680 vec_full_reg_offset(get_field(s, v2)), 1681 vec_full_reg_offset(get_field(s, v3)), 16, 16); 1682 if (get_field(s, m5) & 0x1) { 1683 TCGv_i64 low = tcg_temp_new_i64(); 1684 TCGv_i64 high = tcg_temp_new_i64(); 1685 1686 read_vec_element_i64(high, get_field(s, v1), 0, ES_64); 1687 read_vec_element_i64(low, get_field(s, v1), 1, ES_64); 1688 gen_op_update2_cc_i64(s, CC_OP_VC, low, high); 1689 1690 tcg_temp_free_i64(low); 1691 tcg_temp_free_i64(high); 1692 } 1693 return DISAS_NEXT; 1694} 1695 1696static void gen_clz_i32(TCGv_i32 d, TCGv_i32 a) 1697{ 1698 tcg_gen_clzi_i32(d, a, 32); 1699} 1700 1701static void gen_clz_i64(TCGv_i64 d, TCGv_i64 a) 1702{ 1703 tcg_gen_clzi_i64(d, a, 64); 1704} 1705 1706static DisasJumpType op_vclz(DisasContext *s, DisasOps *o) 1707{ 1708 const uint8_t es = get_field(s, m3); 1709 static const GVecGen2 g[4] = { 1710 { .fno = gen_helper_gvec_vclz8, }, 1711 { .fno = gen_helper_gvec_vclz16, }, 1712 { .fni4 = gen_clz_i32, }, 1713 { .fni8 = gen_clz_i64, }, 1714 }; 1715 1716 if (es > ES_64) { 1717 gen_program_exception(s, PGM_SPECIFICATION); 1718 return DISAS_NORETURN; 1719 } 1720 gen_gvec_2(get_field(s, v1), get_field(s, v2), &g[es]); 1721 return DISAS_NEXT; 1722} 1723 1724static void gen_ctz_i32(TCGv_i32 d, TCGv_i32 a) 1725{ 1726 tcg_gen_ctzi_i32(d, a, 32); 1727} 1728 1729static void gen_ctz_i64(TCGv_i64 d, TCGv_i64 a) 1730{ 1731 tcg_gen_ctzi_i64(d, a, 64); 1732} 1733 1734static DisasJumpType op_vctz(DisasContext *s, DisasOps *o) 1735{ 1736 const uint8_t es = get_field(s, m3); 1737 static const GVecGen2 g[4] = { 1738 { .fno = gen_helper_gvec_vctz8, }, 1739 { .fno = gen_helper_gvec_vctz16, }, 1740 { .fni4 = gen_ctz_i32, }, 1741 { .fni8 = gen_ctz_i64, }, 1742 }; 1743 1744 if (es > ES_64) { 1745 gen_program_exception(s, PGM_SPECIFICATION); 1746 return DISAS_NORETURN; 1747 } 1748 gen_gvec_2(get_field(s, v1), get_field(s, v2), &g[es]); 1749 return DISAS_NEXT; 1750} 1751 1752static DisasJumpType op_vx(DisasContext *s, DisasOps *o) 1753{ 1754 gen_gvec_fn_3(xor, ES_8, get_field(s, v1), get_field(s, v2), 1755 get_field(s, v3)); 1756 return DISAS_NEXT; 1757} 1758 1759static DisasJumpType op_vgfm(DisasContext *s, DisasOps *o) 1760{ 1761 const uint8_t es = get_field(s, m4); 1762 static const GVecGen3 g[4] = { 1763 { .fno = gen_helper_gvec_vgfm8, }, 1764 { .fno = gen_helper_gvec_vgfm16, }, 1765 { .fno = gen_helper_gvec_vgfm32, }, 1766 { .fno = gen_helper_gvec_vgfm64, }, 1767 }; 1768 1769 if (es > ES_64) { 1770 gen_program_exception(s, PGM_SPECIFICATION); 1771 return DISAS_NORETURN; 1772 } 1773 gen_gvec_3(get_field(s, v1), get_field(s, v2), 1774 get_field(s, v3), &g[es]); 1775 return DISAS_NEXT; 1776} 1777 1778static DisasJumpType op_vgfma(DisasContext *s, DisasOps *o) 1779{ 1780 const uint8_t es = get_field(s, m5); 1781 static const GVecGen4 g[4] = { 1782 { .fno = gen_helper_gvec_vgfma8, }, 1783 { .fno = gen_helper_gvec_vgfma16, }, 1784 { .fno = gen_helper_gvec_vgfma32, }, 1785 { .fno = gen_helper_gvec_vgfma64, }, 1786 }; 1787 1788 if (es > ES_64) { 1789 gen_program_exception(s, PGM_SPECIFICATION); 1790 return DISAS_NORETURN; 1791 } 1792 gen_gvec_4(get_field(s, v1), get_field(s, v2), 1793 get_field(s, v3), get_field(s, v4), &g[es]); 1794 return DISAS_NEXT; 1795} 1796 1797static DisasJumpType op_vlc(DisasContext *s, DisasOps *o) 1798{ 1799 const uint8_t es = get_field(s, m3); 1800 1801 if (es > ES_64) { 1802 gen_program_exception(s, PGM_SPECIFICATION); 1803 return DISAS_NORETURN; 1804 } 1805 1806 gen_gvec_fn_2(neg, es, get_field(s, v1), get_field(s, v2)); 1807 return DISAS_NEXT; 1808} 1809 1810static DisasJumpType op_vlp(DisasContext *s, DisasOps *o) 1811{ 1812 const uint8_t es = get_field(s, m3); 1813 1814 if (es > ES_64) { 1815 gen_program_exception(s, PGM_SPECIFICATION); 1816 return DISAS_NORETURN; 1817 } 1818 1819 gen_gvec_fn_2(abs, es, get_field(s, v1), get_field(s, v2)); 1820 return DISAS_NEXT; 1821} 1822 1823static DisasJumpType op_vmx(DisasContext *s, DisasOps *o) 1824{ 1825 const uint8_t v1 = get_field(s, v1); 1826 const uint8_t v2 = get_field(s, v2); 1827 const uint8_t v3 = get_field(s, v3); 1828 const uint8_t es = get_field(s, m4); 1829 1830 if (es > ES_64) { 1831 gen_program_exception(s, PGM_SPECIFICATION); 1832 return DISAS_NORETURN; 1833 } 1834 1835 switch (s->fields.op2) { 1836 case 0xff: 1837 gen_gvec_fn_3(smax, es, v1, v2, v3); 1838 break; 1839 case 0xfd: 1840 gen_gvec_fn_3(umax, es, v1, v2, v3); 1841 break; 1842 case 0xfe: 1843 gen_gvec_fn_3(smin, es, v1, v2, v3); 1844 break; 1845 case 0xfc: 1846 gen_gvec_fn_3(umin, es, v1, v2, v3); 1847 break; 1848 default: 1849 g_assert_not_reached(); 1850 } 1851 return DISAS_NEXT; 1852} 1853 1854static void gen_mal_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b, TCGv_i32 c) 1855{ 1856 TCGv_i32 t0 = tcg_temp_new_i32(); 1857 1858 tcg_gen_mul_i32(t0, a, b); 1859 tcg_gen_add_i32(d, t0, c); 1860 1861 tcg_temp_free_i32(t0); 1862} 1863 1864static void gen_mah_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b, TCGv_i32 c) 1865{ 1866 TCGv_i64 t0 = tcg_temp_new_i64(); 1867 TCGv_i64 t1 = tcg_temp_new_i64(); 1868 TCGv_i64 t2 = tcg_temp_new_i64(); 1869 1870 tcg_gen_ext_i32_i64(t0, a); 1871 tcg_gen_ext_i32_i64(t1, b); 1872 tcg_gen_ext_i32_i64(t2, c); 1873 tcg_gen_mul_i64(t0, t0, t1); 1874 tcg_gen_add_i64(t0, t0, t2); 1875 tcg_gen_extrh_i64_i32(d, t0); 1876 1877 tcg_temp_free(t0); 1878 tcg_temp_free(t1); 1879 tcg_temp_free(t2); 1880} 1881 1882static void gen_malh_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b, TCGv_i32 c) 1883{ 1884 TCGv_i64 t0 = tcg_temp_new_i64(); 1885 TCGv_i64 t1 = tcg_temp_new_i64(); 1886 TCGv_i64 t2 = tcg_temp_new_i64(); 1887 1888 tcg_gen_extu_i32_i64(t0, a); 1889 tcg_gen_extu_i32_i64(t1, b); 1890 tcg_gen_extu_i32_i64(t2, c); 1891 tcg_gen_mul_i64(t0, t0, t1); 1892 tcg_gen_add_i64(t0, t0, t2); 1893 tcg_gen_extrh_i64_i32(d, t0); 1894 1895 tcg_temp_free(t0); 1896 tcg_temp_free(t1); 1897 tcg_temp_free(t2); 1898} 1899 1900static DisasJumpType op_vma(DisasContext *s, DisasOps *o) 1901{ 1902 const uint8_t es = get_field(s, m5); 1903 static const GVecGen4 g_vmal[3] = { 1904 { .fno = gen_helper_gvec_vmal8, }, 1905 { .fno = gen_helper_gvec_vmal16, }, 1906 { .fni4 = gen_mal_i32, }, 1907 }; 1908 static const GVecGen4 g_vmah[3] = { 1909 { .fno = gen_helper_gvec_vmah8, }, 1910 { .fno = gen_helper_gvec_vmah16, }, 1911 { .fni4 = gen_mah_i32, }, 1912 }; 1913 static const GVecGen4 g_vmalh[3] = { 1914 { .fno = gen_helper_gvec_vmalh8, }, 1915 { .fno = gen_helper_gvec_vmalh16, }, 1916 { .fni4 = gen_malh_i32, }, 1917 }; 1918 static const GVecGen4 g_vmae[3] = { 1919 { .fno = gen_helper_gvec_vmae8, }, 1920 { .fno = gen_helper_gvec_vmae16, }, 1921 { .fno = gen_helper_gvec_vmae32, }, 1922 }; 1923 static const GVecGen4 g_vmale[3] = { 1924 { .fno = gen_helper_gvec_vmale8, }, 1925 { .fno = gen_helper_gvec_vmale16, }, 1926 { .fno = gen_helper_gvec_vmale32, }, 1927 }; 1928 static const GVecGen4 g_vmao[3] = { 1929 { .fno = gen_helper_gvec_vmao8, }, 1930 { .fno = gen_helper_gvec_vmao16, }, 1931 { .fno = gen_helper_gvec_vmao32, }, 1932 }; 1933 static const GVecGen4 g_vmalo[3] = { 1934 { .fno = gen_helper_gvec_vmalo8, }, 1935 { .fno = gen_helper_gvec_vmalo16, }, 1936 { .fno = gen_helper_gvec_vmalo32, }, 1937 }; 1938 const GVecGen4 *fn; 1939 1940 if (es > ES_32) { 1941 gen_program_exception(s, PGM_SPECIFICATION); 1942 return DISAS_NORETURN; 1943 } 1944 1945 switch (s->fields.op2) { 1946 case 0xaa: 1947 fn = &g_vmal[es]; 1948 break; 1949 case 0xab: 1950 fn = &g_vmah[es]; 1951 break; 1952 case 0xa9: 1953 fn = &g_vmalh[es]; 1954 break; 1955 case 0xae: 1956 fn = &g_vmae[es]; 1957 break; 1958 case 0xac: 1959 fn = &g_vmale[es]; 1960 break; 1961 case 0xaf: 1962 fn = &g_vmao[es]; 1963 break; 1964 case 0xad: 1965 fn = &g_vmalo[es]; 1966 break; 1967 default: 1968 g_assert_not_reached(); 1969 } 1970 1971 gen_gvec_4(get_field(s, v1), get_field(s, v2), 1972 get_field(s, v3), get_field(s, v4), fn); 1973 return DISAS_NEXT; 1974} 1975 1976static void gen_mh_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 1977{ 1978 TCGv_i32 t = tcg_temp_new_i32(); 1979 1980 tcg_gen_muls2_i32(t, d, a, b); 1981 tcg_temp_free_i32(t); 1982} 1983 1984static void gen_mlh_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 1985{ 1986 TCGv_i32 t = tcg_temp_new_i32(); 1987 1988 tcg_gen_mulu2_i32(t, d, a, b); 1989 tcg_temp_free_i32(t); 1990} 1991 1992static DisasJumpType op_vm(DisasContext *s, DisasOps *o) 1993{ 1994 const uint8_t es = get_field(s, m4); 1995 static const GVecGen3 g_vmh[3] = { 1996 { .fno = gen_helper_gvec_vmh8, }, 1997 { .fno = gen_helper_gvec_vmh16, }, 1998 { .fni4 = gen_mh_i32, }, 1999 }; 2000 static const GVecGen3 g_vmlh[3] = { 2001 { .fno = gen_helper_gvec_vmlh8, }, 2002 { .fno = gen_helper_gvec_vmlh16, }, 2003 { .fni4 = gen_mlh_i32, }, 2004 }; 2005 static const GVecGen3 g_vme[3] = { 2006 { .fno = gen_helper_gvec_vme8, }, 2007 { .fno = gen_helper_gvec_vme16, }, 2008 { .fno = gen_helper_gvec_vme32, }, 2009 }; 2010 static const GVecGen3 g_vmle[3] = { 2011 { .fno = gen_helper_gvec_vmle8, }, 2012 { .fno = gen_helper_gvec_vmle16, }, 2013 { .fno = gen_helper_gvec_vmle32, }, 2014 }; 2015 static const GVecGen3 g_vmo[3] = { 2016 { .fno = gen_helper_gvec_vmo8, }, 2017 { .fno = gen_helper_gvec_vmo16, }, 2018 { .fno = gen_helper_gvec_vmo32, }, 2019 }; 2020 static const GVecGen3 g_vmlo[3] = { 2021 { .fno = gen_helper_gvec_vmlo8, }, 2022 { .fno = gen_helper_gvec_vmlo16, }, 2023 { .fno = gen_helper_gvec_vmlo32, }, 2024 }; 2025 const GVecGen3 *fn; 2026 2027 if (es > ES_32) { 2028 gen_program_exception(s, PGM_SPECIFICATION); 2029 return DISAS_NORETURN; 2030 } 2031 2032 switch (s->fields.op2) { 2033 case 0xa2: 2034 gen_gvec_fn_3(mul, es, get_field(s, v1), 2035 get_field(s, v2), get_field(s, v3)); 2036 return DISAS_NEXT; 2037 case 0xa3: 2038 fn = &g_vmh[es]; 2039 break; 2040 case 0xa1: 2041 fn = &g_vmlh[es]; 2042 break; 2043 case 0xa6: 2044 fn = &g_vme[es]; 2045 break; 2046 case 0xa4: 2047 fn = &g_vmle[es]; 2048 break; 2049 case 0xa7: 2050 fn = &g_vmo[es]; 2051 break; 2052 case 0xa5: 2053 fn = &g_vmlo[es]; 2054 break; 2055 default: 2056 g_assert_not_reached(); 2057 } 2058 2059 gen_gvec_3(get_field(s, v1), get_field(s, v2), 2060 get_field(s, v3), fn); 2061 return DISAS_NEXT; 2062} 2063 2064static DisasJumpType op_vmsl(DisasContext *s, DisasOps *o) 2065{ 2066 TCGv_i64 l1, h1, l2, h2; 2067 2068 if (get_field(s, m5) != ES_64) { 2069 gen_program_exception(s, PGM_SPECIFICATION); 2070 return DISAS_NORETURN; 2071 } 2072 2073 l1 = tcg_temp_new_i64(); 2074 h1 = tcg_temp_new_i64(); 2075 l2 = tcg_temp_new_i64(); 2076 h2 = tcg_temp_new_i64(); 2077 2078 /* Multipy both even elements from v2 and v3 */ 2079 read_vec_element_i64(l1, get_field(s, v2), 0, ES_64); 2080 read_vec_element_i64(h1, get_field(s, v3), 0, ES_64); 2081 tcg_gen_mulu2_i64(l1, h1, l1, h1); 2082 /* Shift result left by one (x2) if requested */ 2083 if (extract32(get_field(s, m6), 3, 1)) { 2084 tcg_gen_add2_i64(l1, h1, l1, h1, l1, h1); 2085 } 2086 2087 /* Multipy both odd elements from v2 and v3 */ 2088 read_vec_element_i64(l2, get_field(s, v2), 1, ES_64); 2089 read_vec_element_i64(h2, get_field(s, v3), 1, ES_64); 2090 tcg_gen_mulu2_i64(l2, h2, l2, h2); 2091 /* Shift result left by one (x2) if requested */ 2092 if (extract32(get_field(s, m6), 2, 1)) { 2093 tcg_gen_add2_i64(l2, h2, l2, h2, l2, h2); 2094 } 2095 2096 /* Add both intermediate results */ 2097 tcg_gen_add2_i64(l1, h1, l1, h1, l2, h2); 2098 /* Add whole v4 */ 2099 read_vec_element_i64(h2, get_field(s, v4), 0, ES_64); 2100 read_vec_element_i64(l2, get_field(s, v4), 1, ES_64); 2101 tcg_gen_add2_i64(l1, h1, l1, h1, l2, h2); 2102 2103 /* Store final result into v1. */ 2104 write_vec_element_i64(h1, get_field(s, v1), 0, ES_64); 2105 write_vec_element_i64(l1, get_field(s, v1), 1, ES_64); 2106 2107 tcg_temp_free_i64(l1); 2108 tcg_temp_free_i64(h1); 2109 tcg_temp_free_i64(l2); 2110 tcg_temp_free_i64(h2); 2111 return DISAS_NEXT; 2112} 2113 2114static DisasJumpType op_vnn(DisasContext *s, DisasOps *o) 2115{ 2116 gen_gvec_fn_3(nand, ES_8, get_field(s, v1), 2117 get_field(s, v2), get_field(s, v3)); 2118 return DISAS_NEXT; 2119} 2120 2121static DisasJumpType op_vno(DisasContext *s, DisasOps *o) 2122{ 2123 gen_gvec_fn_3(nor, ES_8, get_field(s, v1), get_field(s, v2), 2124 get_field(s, v3)); 2125 return DISAS_NEXT; 2126} 2127 2128static DisasJumpType op_vnx(DisasContext *s, DisasOps *o) 2129{ 2130 gen_gvec_fn_3(eqv, ES_8, get_field(s, v1), get_field(s, v2), 2131 get_field(s, v3)); 2132 return DISAS_NEXT; 2133} 2134 2135static DisasJumpType op_vo(DisasContext *s, DisasOps *o) 2136{ 2137 gen_gvec_fn_3(or, ES_8, get_field(s, v1), get_field(s, v2), 2138 get_field(s, v3)); 2139 return DISAS_NEXT; 2140} 2141 2142static DisasJumpType op_voc(DisasContext *s, DisasOps *o) 2143{ 2144 gen_gvec_fn_3(orc, ES_8, get_field(s, v1), get_field(s, v2), 2145 get_field(s, v3)); 2146 return DISAS_NEXT; 2147} 2148 2149static DisasJumpType op_vpopct(DisasContext *s, DisasOps *o) 2150{ 2151 const uint8_t es = get_field(s, m3); 2152 static const GVecGen2 g[4] = { 2153 { .fno = gen_helper_gvec_vpopct8, }, 2154 { .fno = gen_helper_gvec_vpopct16, }, 2155 { .fni4 = tcg_gen_ctpop_i32, }, 2156 { .fni8 = tcg_gen_ctpop_i64, }, 2157 }; 2158 2159 if (es > ES_64 || (es != ES_8 && !s390_has_feat(S390_FEAT_VECTOR_ENH))) { 2160 gen_program_exception(s, PGM_SPECIFICATION); 2161 return DISAS_NORETURN; 2162 } 2163 2164 gen_gvec_2(get_field(s, v1), get_field(s, v2), &g[es]); 2165 return DISAS_NEXT; 2166} 2167 2168static void gen_rim_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b, int32_t c) 2169{ 2170 TCGv_i32 t = tcg_temp_new_i32(); 2171 2172 tcg_gen_rotli_i32(t, a, c & 31); 2173 tcg_gen_and_i32(t, t, b); 2174 tcg_gen_andc_i32(d, d, b); 2175 tcg_gen_or_i32(d, d, t); 2176 2177 tcg_temp_free_i32(t); 2178} 2179 2180static void gen_rim_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b, int64_t c) 2181{ 2182 TCGv_i64 t = tcg_temp_new_i64(); 2183 2184 tcg_gen_rotli_i64(t, a, c & 63); 2185 tcg_gen_and_i64(t, t, b); 2186 tcg_gen_andc_i64(d, d, b); 2187 tcg_gen_or_i64(d, d, t); 2188 2189 tcg_temp_free_i64(t); 2190} 2191 2192static DisasJumpType op_verim(DisasContext *s, DisasOps *o) 2193{ 2194 const uint8_t es = get_field(s, m5); 2195 const uint8_t i4 = get_field(s, i4) & 2196 (NUM_VEC_ELEMENT_BITS(es) - 1); 2197 static const GVecGen3i g[4] = { 2198 { .fno = gen_helper_gvec_verim8, }, 2199 { .fno = gen_helper_gvec_verim16, }, 2200 { .fni4 = gen_rim_i32, 2201 .load_dest = true, }, 2202 { .fni8 = gen_rim_i64, 2203 .load_dest = true, }, 2204 }; 2205 2206 if (es > ES_64) { 2207 gen_program_exception(s, PGM_SPECIFICATION); 2208 return DISAS_NORETURN; 2209 } 2210 2211 gen_gvec_3i(get_field(s, v1), get_field(s, v2), 2212 get_field(s, v3), i4, &g[es]); 2213 return DISAS_NEXT; 2214} 2215 2216static DisasJumpType op_vesv(DisasContext *s, DisasOps *o) 2217{ 2218 const uint8_t es = get_field(s, m4); 2219 const uint8_t v1 = get_field(s, v1); 2220 const uint8_t v2 = get_field(s, v2); 2221 const uint8_t v3 = get_field(s, v3); 2222 2223 if (es > ES_64) { 2224 gen_program_exception(s, PGM_SPECIFICATION); 2225 return DISAS_NORETURN; 2226 } 2227 2228 switch (s->fields.op2) { 2229 case 0x70: 2230 gen_gvec_fn_3(shlv, es, v1, v2, v3); 2231 break; 2232 case 0x73: 2233 gen_gvec_fn_3(rotlv, es, v1, v2, v3); 2234 break; 2235 case 0x7a: 2236 gen_gvec_fn_3(sarv, es, v1, v2, v3); 2237 break; 2238 case 0x78: 2239 gen_gvec_fn_3(shrv, es, v1, v2, v3); 2240 break; 2241 default: 2242 g_assert_not_reached(); 2243 } 2244 return DISAS_NEXT; 2245} 2246 2247static DisasJumpType op_ves(DisasContext *s, DisasOps *o) 2248{ 2249 const uint8_t es = get_field(s, m4); 2250 const uint8_t d2 = get_field(s, d2) & 2251 (NUM_VEC_ELEMENT_BITS(es) - 1); 2252 const uint8_t v1 = get_field(s, v1); 2253 const uint8_t v3 = get_field(s, v3); 2254 TCGv_i32 shift; 2255 2256 if (es > ES_64) { 2257 gen_program_exception(s, PGM_SPECIFICATION); 2258 return DISAS_NORETURN; 2259 } 2260 2261 if (likely(!get_field(s, b2))) { 2262 switch (s->fields.op2) { 2263 case 0x30: 2264 gen_gvec_fn_2i(shli, es, v1, v3, d2); 2265 break; 2266 case 0x33: 2267 gen_gvec_fn_2i(rotli, es, v1, v3, d2); 2268 break; 2269 case 0x3a: 2270 gen_gvec_fn_2i(sari, es, v1, v3, d2); 2271 break; 2272 case 0x38: 2273 gen_gvec_fn_2i(shri, es, v1, v3, d2); 2274 break; 2275 default: 2276 g_assert_not_reached(); 2277 } 2278 } else { 2279 shift = tcg_temp_new_i32(); 2280 tcg_gen_extrl_i64_i32(shift, o->addr1); 2281 tcg_gen_andi_i32(shift, shift, NUM_VEC_ELEMENT_BITS(es) - 1); 2282 switch (s->fields.op2) { 2283 case 0x30: 2284 gen_gvec_fn_2s(shls, es, v1, v3, shift); 2285 break; 2286 case 0x33: 2287 gen_gvec_fn_2s(rotls, es, v1, v3, shift); 2288 break; 2289 case 0x3a: 2290 gen_gvec_fn_2s(sars, es, v1, v3, shift); 2291 break; 2292 case 0x38: 2293 gen_gvec_fn_2s(shrs, es, v1, v3, shift); 2294 break; 2295 default: 2296 g_assert_not_reached(); 2297 } 2298 tcg_temp_free_i32(shift); 2299 } 2300 return DISAS_NEXT; 2301} 2302 2303static DisasJumpType gen_vsh_by_byte(DisasContext *s, DisasOps *o, 2304 gen_helper_gvec_2i *gen, 2305 gen_helper_gvec_3 *gen_ve2) 2306{ 2307 bool byte = s->insn->data; 2308 2309 if (!byte && s390_has_feat(S390_FEAT_VECTOR_ENH2)) { 2310 gen_gvec_3_ool(get_field(s, v1), get_field(s, v2), 2311 get_field(s, v3), 0, gen_ve2); 2312 } else { 2313 TCGv_i64 shift = tcg_temp_new_i64(); 2314 2315 read_vec_element_i64(shift, get_field(s, v3), 7, ES_8); 2316 tcg_gen_andi_i64(shift, shift, byte ? 0x78 : 7); 2317 gen_gvec_2i_ool(get_field(s, v1), get_field(s, v2), shift, 0, gen); 2318 tcg_temp_free_i64(shift); 2319 } 2320 return DISAS_NEXT; 2321} 2322 2323static DisasJumpType op_vsl(DisasContext *s, DisasOps *o) 2324{ 2325 return gen_vsh_by_byte(s, o, gen_helper_gvec_vsl, 2326 gen_helper_gvec_vsl_ve2); 2327} 2328 2329static DisasJumpType op_vsra(DisasContext *s, DisasOps *o) 2330{ 2331 return gen_vsh_by_byte(s, o, gen_helper_gvec_vsra, 2332 gen_helper_gvec_vsra_ve2); 2333} 2334 2335static DisasJumpType op_vsrl(DisasContext *s, DisasOps *o) 2336{ 2337 return gen_vsh_by_byte(s, o, gen_helper_gvec_vsrl, 2338 gen_helper_gvec_vsrl_ve2); 2339} 2340 2341static DisasJumpType op_vsld(DisasContext *s, DisasOps *o) 2342{ 2343 const bool byte = s->insn->data; 2344 const uint8_t mask = byte ? 15 : 7; 2345 const uint8_t mul = byte ? 8 : 1; 2346 const uint8_t i4 = get_field(s, i4); 2347 const int right_shift = 64 - (i4 & 7) * mul; 2348 TCGv_i64 t0, t1, t2; 2349 2350 if (i4 & ~mask) { 2351 gen_program_exception(s, PGM_SPECIFICATION); 2352 return DISAS_NORETURN; 2353 } 2354 2355 t0 = tcg_temp_new_i64(); 2356 t1 = tcg_temp_new_i64(); 2357 t2 = tcg_temp_new_i64(); 2358 2359 if ((i4 & 8) == 0) { 2360 read_vec_element_i64(t0, get_field(s, v2), 0, ES_64); 2361 read_vec_element_i64(t1, get_field(s, v2), 1, ES_64); 2362 read_vec_element_i64(t2, get_field(s, v3), 0, ES_64); 2363 } else { 2364 read_vec_element_i64(t0, get_field(s, v2), 1, ES_64); 2365 read_vec_element_i64(t1, get_field(s, v3), 0, ES_64); 2366 read_vec_element_i64(t2, get_field(s, v3), 1, ES_64); 2367 } 2368 2369 tcg_gen_extract2_i64(t0, t1, t0, right_shift); 2370 tcg_gen_extract2_i64(t1, t2, t1, right_shift); 2371 2372 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 2373 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 2374 2375 tcg_temp_free(t0); 2376 tcg_temp_free(t1); 2377 tcg_temp_free(t2); 2378 return DISAS_NEXT; 2379} 2380 2381static DisasJumpType op_vsrd(DisasContext *s, DisasOps *o) 2382{ 2383 const uint8_t i4 = get_field(s, i4); 2384 TCGv_i64 t0, t1, t2; 2385 2386 if (i4 & ~7) { 2387 gen_program_exception(s, PGM_SPECIFICATION); 2388 return DISAS_NORETURN; 2389 } 2390 2391 t0 = tcg_temp_new_i64(); 2392 t1 = tcg_temp_new_i64(); 2393 t2 = tcg_temp_new_i64(); 2394 2395 read_vec_element_i64(t0, get_field(s, v2), 1, ES_64); 2396 read_vec_element_i64(t1, get_field(s, v3), 0, ES_64); 2397 read_vec_element_i64(t2, get_field(s, v3), 1, ES_64); 2398 2399 tcg_gen_extract2_i64(t0, t1, t0, i4); 2400 tcg_gen_extract2_i64(t1, t2, t1, i4); 2401 2402 write_vec_element_i64(t0, get_field(s, v1), 0, ES_64); 2403 write_vec_element_i64(t1, get_field(s, v1), 1, ES_64); 2404 2405 tcg_temp_free(t0); 2406 tcg_temp_free(t1); 2407 tcg_temp_free(t2); 2408 return DISAS_NEXT; 2409} 2410 2411static DisasJumpType op_vs(DisasContext *s, DisasOps *o) 2412{ 2413 const uint8_t es = get_field(s, m4); 2414 2415 if (es > ES_128) { 2416 gen_program_exception(s, PGM_SPECIFICATION); 2417 return DISAS_NORETURN; 2418 } else if (es == ES_128) { 2419 gen_gvec128_3_i64(tcg_gen_sub2_i64, get_field(s, v1), 2420 get_field(s, v2), get_field(s, v3)); 2421 return DISAS_NEXT; 2422 } 2423 gen_gvec_fn_3(sub, es, get_field(s, v1), get_field(s, v2), 2424 get_field(s, v3)); 2425 return DISAS_NEXT; 2426} 2427 2428static void gen_scbi_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b) 2429{ 2430 tcg_gen_setcond_i32(TCG_COND_GEU, d, a, b); 2431} 2432 2433static void gen_scbi_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b) 2434{ 2435 tcg_gen_setcond_i64(TCG_COND_GEU, d, a, b); 2436} 2437 2438static void gen_scbi2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, 2439 TCGv_i64 ah, TCGv_i64 bl, TCGv_i64 bh) 2440{ 2441 TCGv_i64 th = tcg_temp_new_i64(); 2442 TCGv_i64 tl = tcg_temp_new_i64(); 2443 TCGv_i64 zero = tcg_const_i64(0); 2444 2445 tcg_gen_sub2_i64(tl, th, al, zero, bl, zero); 2446 tcg_gen_andi_i64(th, th, 1); 2447 tcg_gen_sub2_i64(tl, th, ah, zero, th, zero); 2448 tcg_gen_sub2_i64(tl, th, tl, th, bh, zero); 2449 /* "invert" the result: -1 -> 0; 0 -> 1 */ 2450 tcg_gen_addi_i64(dl, th, 1); 2451 tcg_gen_mov_i64(dh, zero); 2452 2453 tcg_temp_free_i64(th); 2454 tcg_temp_free_i64(tl); 2455 tcg_temp_free_i64(zero); 2456} 2457 2458static DisasJumpType op_vscbi(DisasContext *s, DisasOps *o) 2459{ 2460 const uint8_t es = get_field(s, m4); 2461 static const GVecGen3 g[4] = { 2462 { .fno = gen_helper_gvec_vscbi8, }, 2463 { .fno = gen_helper_gvec_vscbi16, }, 2464 { .fni4 = gen_scbi_i32, }, 2465 { .fni8 = gen_scbi_i64, }, 2466 }; 2467 2468 if (es > ES_128) { 2469 gen_program_exception(s, PGM_SPECIFICATION); 2470 return DISAS_NORETURN; 2471 } else if (es == ES_128) { 2472 gen_gvec128_3_i64(gen_scbi2_i64, get_field(s, v1), 2473 get_field(s, v2), get_field(s, v3)); 2474 return DISAS_NEXT; 2475 } 2476 gen_gvec_3(get_field(s, v1), get_field(s, v2), 2477 get_field(s, v3), &g[es]); 2478 return DISAS_NEXT; 2479} 2480 2481static void gen_sbi2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, TCGv_i64 ah, 2482 TCGv_i64 bl, TCGv_i64 bh, TCGv_i64 cl, TCGv_i64 ch) 2483{ 2484 TCGv_i64 tl = tcg_temp_new_i64(); 2485 TCGv_i64 th = tcg_temp_new_i64(); 2486 2487 tcg_gen_not_i64(tl, bl); 2488 tcg_gen_not_i64(th, bh); 2489 gen_ac2_i64(dl, dh, al, ah, tl, th, cl, ch); 2490 tcg_temp_free_i64(tl); 2491 tcg_temp_free_i64(th); 2492} 2493 2494static DisasJumpType op_vsbi(DisasContext *s, DisasOps *o) 2495{ 2496 if (get_field(s, m5) != ES_128) { 2497 gen_program_exception(s, PGM_SPECIFICATION); 2498 return DISAS_NORETURN; 2499 } 2500 2501 gen_gvec128_4_i64(gen_sbi2_i64, get_field(s, v1), 2502 get_field(s, v2), get_field(s, v3), 2503 get_field(s, v4)); 2504 return DISAS_NEXT; 2505} 2506 2507static void gen_sbcbi2_i64(TCGv_i64 dl, TCGv_i64 dh, TCGv_i64 al, TCGv_i64 ah, 2508 TCGv_i64 bl, TCGv_i64 bh, TCGv_i64 cl, TCGv_i64 ch) 2509{ 2510 TCGv_i64 th = tcg_temp_new_i64(); 2511 TCGv_i64 tl = tcg_temp_new_i64(); 2512 2513 tcg_gen_not_i64(tl, bl); 2514 tcg_gen_not_i64(th, bh); 2515 gen_accc2_i64(dl, dh, al, ah, tl, th, cl, ch); 2516 2517 tcg_temp_free_i64(tl); 2518 tcg_temp_free_i64(th); 2519} 2520 2521static DisasJumpType op_vsbcbi(DisasContext *s, DisasOps *o) 2522{ 2523 if (get_field(s, m5) != ES_128) { 2524 gen_program_exception(s, PGM_SPECIFICATION); 2525 return DISAS_NORETURN; 2526 } 2527 2528 gen_gvec128_4_i64(gen_sbcbi2_i64, get_field(s, v1), 2529 get_field(s, v2), get_field(s, v3), 2530 get_field(s, v4)); 2531 return DISAS_NEXT; 2532} 2533 2534static DisasJumpType op_vsumg(DisasContext *s, DisasOps *o) 2535{ 2536 const uint8_t es = get_field(s, m4); 2537 TCGv_i64 sum, tmp; 2538 uint8_t dst_idx; 2539 2540 if (es == ES_8 || es > ES_32) { 2541 gen_program_exception(s, PGM_SPECIFICATION); 2542 return DISAS_NORETURN; 2543 } 2544 2545 sum = tcg_temp_new_i64(); 2546 tmp = tcg_temp_new_i64(); 2547 for (dst_idx = 0; dst_idx < 2; dst_idx++) { 2548 uint8_t idx = dst_idx * NUM_VEC_ELEMENTS(es) / 2; 2549 const uint8_t max_idx = idx + NUM_VEC_ELEMENTS(es) / 2 - 1; 2550 2551 read_vec_element_i64(sum, get_field(s, v3), max_idx, es); 2552 for (; idx <= max_idx; idx++) { 2553 read_vec_element_i64(tmp, get_field(s, v2), idx, es); 2554 tcg_gen_add_i64(sum, sum, tmp); 2555 } 2556 write_vec_element_i64(sum, get_field(s, v1), dst_idx, ES_64); 2557 } 2558 tcg_temp_free_i64(sum); 2559 tcg_temp_free_i64(tmp); 2560 return DISAS_NEXT; 2561} 2562 2563static DisasJumpType op_vsumq(DisasContext *s, DisasOps *o) 2564{ 2565 const uint8_t es = get_field(s, m4); 2566 const uint8_t max_idx = NUM_VEC_ELEMENTS(es) - 1; 2567 TCGv_i64 sumh, suml, zero, tmpl; 2568 uint8_t idx; 2569 2570 if (es < ES_32 || es > ES_64) { 2571 gen_program_exception(s, PGM_SPECIFICATION); 2572 return DISAS_NORETURN; 2573 } 2574 2575 sumh = tcg_const_i64(0); 2576 suml = tcg_temp_new_i64(); 2577 zero = tcg_const_i64(0); 2578 tmpl = tcg_temp_new_i64(); 2579 2580 read_vec_element_i64(suml, get_field(s, v3), max_idx, es); 2581 for (idx = 0; idx <= max_idx; idx++) { 2582 read_vec_element_i64(tmpl, get_field(s, v2), idx, es); 2583 tcg_gen_add2_i64(suml, sumh, suml, sumh, tmpl, zero); 2584 } 2585 write_vec_element_i64(sumh, get_field(s, v1), 0, ES_64); 2586 write_vec_element_i64(suml, get_field(s, v1), 1, ES_64); 2587 2588 tcg_temp_free_i64(sumh); 2589 tcg_temp_free_i64(suml); 2590 tcg_temp_free_i64(zero); 2591 tcg_temp_free_i64(tmpl); 2592 return DISAS_NEXT; 2593} 2594 2595static DisasJumpType op_vsum(DisasContext *s, DisasOps *o) 2596{ 2597 const uint8_t es = get_field(s, m4); 2598 TCGv_i32 sum, tmp; 2599 uint8_t dst_idx; 2600 2601 if (es > ES_16) { 2602 gen_program_exception(s, PGM_SPECIFICATION); 2603 return DISAS_NORETURN; 2604 } 2605 2606 sum = tcg_temp_new_i32(); 2607 tmp = tcg_temp_new_i32(); 2608 for (dst_idx = 0; dst_idx < 4; dst_idx++) { 2609 uint8_t idx = dst_idx * NUM_VEC_ELEMENTS(es) / 4; 2610 const uint8_t max_idx = idx + NUM_VEC_ELEMENTS(es) / 4 - 1; 2611 2612 read_vec_element_i32(sum, get_field(s, v3), max_idx, es); 2613 for (; idx <= max_idx; idx++) { 2614 read_vec_element_i32(tmp, get_field(s, v2), idx, es); 2615 tcg_gen_add_i32(sum, sum, tmp); 2616 } 2617 write_vec_element_i32(sum, get_field(s, v1), dst_idx, ES_32); 2618 } 2619 tcg_temp_free_i32(sum); 2620 tcg_temp_free_i32(tmp); 2621 return DISAS_NEXT; 2622} 2623 2624static DisasJumpType op_vtm(DisasContext *s, DisasOps *o) 2625{ 2626 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), 2627 cpu_env, 0, gen_helper_gvec_vtm); 2628 set_cc_static(s); 2629 return DISAS_NEXT; 2630} 2631 2632static DisasJumpType op_vfae(DisasContext *s, DisasOps *o) 2633{ 2634 const uint8_t es = get_field(s, m4); 2635 const uint8_t m5 = get_field(s, m5); 2636 static gen_helper_gvec_3 * const g[3] = { 2637 gen_helper_gvec_vfae8, 2638 gen_helper_gvec_vfae16, 2639 gen_helper_gvec_vfae32, 2640 }; 2641 static gen_helper_gvec_3_ptr * const g_cc[3] = { 2642 gen_helper_gvec_vfae_cc8, 2643 gen_helper_gvec_vfae_cc16, 2644 gen_helper_gvec_vfae_cc32, 2645 }; 2646 if (es > ES_32) { 2647 gen_program_exception(s, PGM_SPECIFICATION); 2648 return DISAS_NORETURN; 2649 } 2650 2651 if (extract32(m5, 0, 1)) { 2652 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), 2653 get_field(s, v3), cpu_env, m5, g_cc[es]); 2654 set_cc_static(s); 2655 } else { 2656 gen_gvec_3_ool(get_field(s, v1), get_field(s, v2), 2657 get_field(s, v3), m5, g[es]); 2658 } 2659 return DISAS_NEXT; 2660} 2661 2662static DisasJumpType op_vfee(DisasContext *s, DisasOps *o) 2663{ 2664 const uint8_t es = get_field(s, m4); 2665 const uint8_t m5 = get_field(s, m5); 2666 static gen_helper_gvec_3 * const g[3] = { 2667 gen_helper_gvec_vfee8, 2668 gen_helper_gvec_vfee16, 2669 gen_helper_gvec_vfee32, 2670 }; 2671 static gen_helper_gvec_3_ptr * const g_cc[3] = { 2672 gen_helper_gvec_vfee_cc8, 2673 gen_helper_gvec_vfee_cc16, 2674 gen_helper_gvec_vfee_cc32, 2675 }; 2676 2677 if (es > ES_32 || m5 & ~0x3) { 2678 gen_program_exception(s, PGM_SPECIFICATION); 2679 return DISAS_NORETURN; 2680 } 2681 2682 if (extract32(m5, 0, 1)) { 2683 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), 2684 get_field(s, v3), cpu_env, m5, g_cc[es]); 2685 set_cc_static(s); 2686 } else { 2687 gen_gvec_3_ool(get_field(s, v1), get_field(s, v2), 2688 get_field(s, v3), m5, g[es]); 2689 } 2690 return DISAS_NEXT; 2691} 2692 2693static DisasJumpType op_vfene(DisasContext *s, DisasOps *o) 2694{ 2695 const uint8_t es = get_field(s, m4); 2696 const uint8_t m5 = get_field(s, m5); 2697 static gen_helper_gvec_3 * const g[3] = { 2698 gen_helper_gvec_vfene8, 2699 gen_helper_gvec_vfene16, 2700 gen_helper_gvec_vfene32, 2701 }; 2702 static gen_helper_gvec_3_ptr * const g_cc[3] = { 2703 gen_helper_gvec_vfene_cc8, 2704 gen_helper_gvec_vfene_cc16, 2705 gen_helper_gvec_vfene_cc32, 2706 }; 2707 2708 if (es > ES_32 || m5 & ~0x3) { 2709 gen_program_exception(s, PGM_SPECIFICATION); 2710 return DISAS_NORETURN; 2711 } 2712 2713 if (extract32(m5, 0, 1)) { 2714 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), 2715 get_field(s, v3), cpu_env, m5, g_cc[es]); 2716 set_cc_static(s); 2717 } else { 2718 gen_gvec_3_ool(get_field(s, v1), get_field(s, v2), 2719 get_field(s, v3), m5, g[es]); 2720 } 2721 return DISAS_NEXT; 2722} 2723 2724static DisasJumpType op_vistr(DisasContext *s, DisasOps *o) 2725{ 2726 const uint8_t es = get_field(s, m4); 2727 const uint8_t m5 = get_field(s, m5); 2728 static gen_helper_gvec_2 * const g[3] = { 2729 gen_helper_gvec_vistr8, 2730 gen_helper_gvec_vistr16, 2731 gen_helper_gvec_vistr32, 2732 }; 2733 static gen_helper_gvec_2_ptr * const g_cc[3] = { 2734 gen_helper_gvec_vistr_cc8, 2735 gen_helper_gvec_vistr_cc16, 2736 gen_helper_gvec_vistr_cc32, 2737 }; 2738 2739 if (es > ES_32 || m5 & ~0x1) { 2740 gen_program_exception(s, PGM_SPECIFICATION); 2741 return DISAS_NORETURN; 2742 } 2743 2744 if (extract32(m5, 0, 1)) { 2745 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), 2746 cpu_env, 0, g_cc[es]); 2747 set_cc_static(s); 2748 } else { 2749 gen_gvec_2_ool(get_field(s, v1), get_field(s, v2), 0, 2750 g[es]); 2751 } 2752 return DISAS_NEXT; 2753} 2754 2755static DisasJumpType op_vstrc(DisasContext *s, DisasOps *o) 2756{ 2757 const uint8_t es = get_field(s, m5); 2758 const uint8_t m6 = get_field(s, m6); 2759 static gen_helper_gvec_4 * const g[3] = { 2760 gen_helper_gvec_vstrc8, 2761 gen_helper_gvec_vstrc16, 2762 gen_helper_gvec_vstrc32, 2763 }; 2764 static gen_helper_gvec_4 * const g_rt[3] = { 2765 gen_helper_gvec_vstrc_rt8, 2766 gen_helper_gvec_vstrc_rt16, 2767 gen_helper_gvec_vstrc_rt32, 2768 }; 2769 static gen_helper_gvec_4_ptr * const g_cc[3] = { 2770 gen_helper_gvec_vstrc_cc8, 2771 gen_helper_gvec_vstrc_cc16, 2772 gen_helper_gvec_vstrc_cc32, 2773 }; 2774 static gen_helper_gvec_4_ptr * const g_cc_rt[3] = { 2775 gen_helper_gvec_vstrc_cc_rt8, 2776 gen_helper_gvec_vstrc_cc_rt16, 2777 gen_helper_gvec_vstrc_cc_rt32, 2778 }; 2779 2780 if (es > ES_32) { 2781 gen_program_exception(s, PGM_SPECIFICATION); 2782 return DISAS_NORETURN; 2783 } 2784 2785 if (extract32(m6, 0, 1)) { 2786 if (extract32(m6, 2, 1)) { 2787 gen_gvec_4_ptr(get_field(s, v1), get_field(s, v2), 2788 get_field(s, v3), get_field(s, v4), 2789 cpu_env, m6, g_cc_rt[es]); 2790 } else { 2791 gen_gvec_4_ptr(get_field(s, v1), get_field(s, v2), 2792 get_field(s, v3), get_field(s, v4), 2793 cpu_env, m6, g_cc[es]); 2794 } 2795 set_cc_static(s); 2796 } else { 2797 if (extract32(m6, 2, 1)) { 2798 gen_gvec_4_ool(get_field(s, v1), get_field(s, v2), 2799 get_field(s, v3), get_field(s, v4), 2800 m6, g_rt[es]); 2801 } else { 2802 gen_gvec_4_ool(get_field(s, v1), get_field(s, v2), 2803 get_field(s, v3), get_field(s, v4), 2804 m6, g[es]); 2805 } 2806 } 2807 return DISAS_NEXT; 2808} 2809 2810static DisasJumpType op_vstrs(DisasContext *s, DisasOps *o) 2811{ 2812 typedef void (*helper_vstrs)(TCGv_ptr, TCGv_ptr, TCGv_ptr, 2813 TCGv_ptr, TCGv_ptr, TCGv_i32); 2814 static const helper_vstrs fns[3][2] = { 2815 { gen_helper_gvec_vstrs_8, gen_helper_gvec_vstrs_zs8 }, 2816 { gen_helper_gvec_vstrs_16, gen_helper_gvec_vstrs_zs16 }, 2817 { gen_helper_gvec_vstrs_32, gen_helper_gvec_vstrs_zs32 }, 2818 }; 2819 const uint8_t es = get_field(s, m5); 2820 const uint8_t m6 = get_field(s, m6); 2821 const bool zs = extract32(m6, 1, 1); 2822 2823 if (es > ES_32 || m6 & ~2) { 2824 gen_program_exception(s, PGM_SPECIFICATION); 2825 return DISAS_NORETURN; 2826 } 2827 2828 gen_gvec_4_ptr(get_field(s, v1), get_field(s, v2), 2829 get_field(s, v3), get_field(s, v4), 2830 cpu_env, 0, fns[es][zs]); 2831 set_cc_static(s); 2832 return DISAS_NEXT; 2833} 2834 2835static DisasJumpType op_vfa(DisasContext *s, DisasOps *o) 2836{ 2837 const uint8_t fpf = get_field(s, m4); 2838 const uint8_t m5 = get_field(s, m5); 2839 gen_helper_gvec_3_ptr *fn = NULL; 2840 2841 switch (s->fields.op2) { 2842 case 0xe3: 2843 switch (fpf) { 2844 case FPF_SHORT: 2845 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2846 fn = gen_helper_gvec_vfa32; 2847 } 2848 break; 2849 case FPF_LONG: 2850 fn = gen_helper_gvec_vfa64; 2851 break; 2852 case FPF_EXT: 2853 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2854 fn = gen_helper_gvec_vfa128; 2855 } 2856 break; 2857 default: 2858 break; 2859 } 2860 break; 2861 case 0xe5: 2862 switch (fpf) { 2863 case FPF_SHORT: 2864 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2865 fn = gen_helper_gvec_vfd32; 2866 } 2867 break; 2868 case FPF_LONG: 2869 fn = gen_helper_gvec_vfd64; 2870 break; 2871 case FPF_EXT: 2872 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2873 fn = gen_helper_gvec_vfd128; 2874 } 2875 break; 2876 default: 2877 break; 2878 } 2879 break; 2880 case 0xe7: 2881 switch (fpf) { 2882 case FPF_SHORT: 2883 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2884 fn = gen_helper_gvec_vfm32; 2885 } 2886 break; 2887 case FPF_LONG: 2888 fn = gen_helper_gvec_vfm64; 2889 break; 2890 case FPF_EXT: 2891 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2892 fn = gen_helper_gvec_vfm128; 2893 } 2894 break; 2895 default: 2896 break; 2897 } 2898 break; 2899 case 0xe2: 2900 switch (fpf) { 2901 case FPF_SHORT: 2902 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2903 fn = gen_helper_gvec_vfs32; 2904 } 2905 break; 2906 case FPF_LONG: 2907 fn = gen_helper_gvec_vfs64; 2908 break; 2909 case FPF_EXT: 2910 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2911 fn = gen_helper_gvec_vfs128; 2912 } 2913 break; 2914 default: 2915 break; 2916 } 2917 break; 2918 default: 2919 g_assert_not_reached(); 2920 } 2921 2922 if (!fn || extract32(m5, 0, 3)) { 2923 gen_program_exception(s, PGM_SPECIFICATION); 2924 return DISAS_NORETURN; 2925 } 2926 2927 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), 2928 get_field(s, v3), cpu_env, m5, fn); 2929 return DISAS_NEXT; 2930} 2931 2932static DisasJumpType op_wfc(DisasContext *s, DisasOps *o) 2933{ 2934 const uint8_t fpf = get_field(s, m3); 2935 const uint8_t m4 = get_field(s, m4); 2936 gen_helper_gvec_2_ptr *fn = NULL; 2937 2938 switch (fpf) { 2939 case FPF_SHORT: 2940 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2941 fn = gen_helper_gvec_wfk32; 2942 if (s->fields.op2 == 0xcb) { 2943 fn = gen_helper_gvec_wfc32; 2944 } 2945 } 2946 break; 2947 case FPF_LONG: 2948 fn = gen_helper_gvec_wfk64; 2949 if (s->fields.op2 == 0xcb) { 2950 fn = gen_helper_gvec_wfc64; 2951 } 2952 break; 2953 case FPF_EXT: 2954 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 2955 fn = gen_helper_gvec_wfk128; 2956 if (s->fields.op2 == 0xcb) { 2957 fn = gen_helper_gvec_wfc128; 2958 } 2959 } 2960 break; 2961 default: 2962 break; 2963 }; 2964 2965 if (!fn || m4) { 2966 gen_program_exception(s, PGM_SPECIFICATION); 2967 return DISAS_NORETURN; 2968 } 2969 2970 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), cpu_env, 0, fn); 2971 set_cc_static(s); 2972 return DISAS_NEXT; 2973} 2974 2975static DisasJumpType op_vfc(DisasContext *s, DisasOps *o) 2976{ 2977 const uint8_t fpf = get_field(s, m4); 2978 const uint8_t m5 = get_field(s, m5); 2979 const uint8_t m6 = get_field(s, m6); 2980 const bool cs = extract32(m6, 0, 1); 2981 const bool sq = extract32(m5, 2, 1); 2982 gen_helper_gvec_3_ptr *fn = NULL; 2983 2984 switch (s->fields.op2) { 2985 case 0xe8: 2986 switch (fpf) { 2987 case FPF_SHORT: 2988 fn = cs ? gen_helper_gvec_vfce32_cc : gen_helper_gvec_vfce32; 2989 break; 2990 case FPF_LONG: 2991 fn = cs ? gen_helper_gvec_vfce64_cc : gen_helper_gvec_vfce64; 2992 break; 2993 case FPF_EXT: 2994 fn = cs ? gen_helper_gvec_vfce128_cc : gen_helper_gvec_vfce128; 2995 break; 2996 default: 2997 break; 2998 } 2999 break; 3000 case 0xeb: 3001 switch (fpf) { 3002 case FPF_SHORT: 3003 fn = cs ? gen_helper_gvec_vfch32_cc : gen_helper_gvec_vfch32; 3004 break; 3005 case FPF_LONG: 3006 fn = cs ? gen_helper_gvec_vfch64_cc : gen_helper_gvec_vfch64; 3007 break; 3008 case FPF_EXT: 3009 fn = cs ? gen_helper_gvec_vfch128_cc : gen_helper_gvec_vfch128; 3010 break; 3011 default: 3012 break; 3013 } 3014 break; 3015 case 0xea: 3016 switch (fpf) { 3017 case FPF_SHORT: 3018 fn = cs ? gen_helper_gvec_vfche32_cc : gen_helper_gvec_vfche32; 3019 break; 3020 case FPF_LONG: 3021 fn = cs ? gen_helper_gvec_vfche64_cc : gen_helper_gvec_vfche64; 3022 break; 3023 case FPF_EXT: 3024 fn = cs ? gen_helper_gvec_vfche128_cc : gen_helper_gvec_vfche128; 3025 break; 3026 default: 3027 break; 3028 } 3029 break; 3030 default: 3031 g_assert_not_reached(); 3032 } 3033 3034 if (!fn || extract32(m5, 0, 2) || extract32(m6, 1, 3) || 3035 (!s390_has_feat(S390_FEAT_VECTOR_ENH) && (fpf != FPF_LONG || sq))) { 3036 gen_program_exception(s, PGM_SPECIFICATION); 3037 return DISAS_NORETURN; 3038 } 3039 3040 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), get_field(s, v3), 3041 cpu_env, m5, fn); 3042 if (cs) { 3043 set_cc_static(s); 3044 } 3045 return DISAS_NEXT; 3046} 3047 3048static DisasJumpType op_vcdg(DisasContext *s, DisasOps *o) 3049{ 3050 const uint8_t fpf = get_field(s, m3); 3051 const uint8_t m4 = get_field(s, m4); 3052 const uint8_t erm = get_field(s, m5); 3053 gen_helper_gvec_2_ptr *fn = NULL; 3054 3055 3056 switch (s->fields.op2) { 3057 case 0xc3: 3058 switch (fpf) { 3059 case FPF_LONG: 3060 fn = gen_helper_gvec_vcdg64; 3061 break; 3062 case FPF_SHORT: 3063 if (s390_has_feat(S390_FEAT_VECTOR_ENH2)) { 3064 fn = gen_helper_gvec_vcdg32; 3065 } 3066 break; 3067 default: 3068 break; 3069 } 3070 break; 3071 case 0xc1: 3072 switch (fpf) { 3073 case FPF_LONG: 3074 fn = gen_helper_gvec_vcdlg64; 3075 break; 3076 case FPF_SHORT: 3077 if (s390_has_feat(S390_FEAT_VECTOR_ENH2)) { 3078 fn = gen_helper_gvec_vcdlg32; 3079 } 3080 break; 3081 default: 3082 break; 3083 } 3084 break; 3085 case 0xc2: 3086 switch (fpf) { 3087 case FPF_LONG: 3088 fn = gen_helper_gvec_vcgd64; 3089 break; 3090 case FPF_SHORT: 3091 if (s390_has_feat(S390_FEAT_VECTOR_ENH2)) { 3092 fn = gen_helper_gvec_vcgd32; 3093 } 3094 break; 3095 default: 3096 break; 3097 } 3098 break; 3099 case 0xc0: 3100 switch (fpf) { 3101 case FPF_LONG: 3102 fn = gen_helper_gvec_vclgd64; 3103 break; 3104 case FPF_SHORT: 3105 if (s390_has_feat(S390_FEAT_VECTOR_ENH2)) { 3106 fn = gen_helper_gvec_vclgd32; 3107 } 3108 break; 3109 default: 3110 break; 3111 } 3112 break; 3113 case 0xc7: 3114 switch (fpf) { 3115 case FPF_SHORT: 3116 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3117 fn = gen_helper_gvec_vfi32; 3118 } 3119 break; 3120 case FPF_LONG: 3121 fn = gen_helper_gvec_vfi64; 3122 break; 3123 case FPF_EXT: 3124 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3125 fn = gen_helper_gvec_vfi128; 3126 } 3127 break; 3128 default: 3129 break; 3130 } 3131 break; 3132 case 0xc5: 3133 switch (fpf) { 3134 case FPF_LONG: 3135 fn = gen_helper_gvec_vflr64; 3136 break; 3137 case FPF_EXT: 3138 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3139 fn = gen_helper_gvec_vflr128; 3140 } 3141 break; 3142 default: 3143 break; 3144 } 3145 break; 3146 default: 3147 g_assert_not_reached(); 3148 } 3149 3150 if (!fn || extract32(m4, 0, 2) || erm > 7 || erm == 2) { 3151 gen_program_exception(s, PGM_SPECIFICATION); 3152 return DISAS_NORETURN; 3153 } 3154 3155 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), cpu_env, 3156 deposit32(m4, 4, 4, erm), fn); 3157 return DISAS_NEXT; 3158} 3159 3160static DisasJumpType op_vfll(DisasContext *s, DisasOps *o) 3161{ 3162 const uint8_t fpf = get_field(s, m3); 3163 const uint8_t m4 = get_field(s, m4); 3164 gen_helper_gvec_2_ptr *fn = NULL; 3165 3166 switch (fpf) { 3167 case FPF_SHORT: 3168 fn = gen_helper_gvec_vfll32; 3169 break; 3170 case FPF_LONG: 3171 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3172 fn = gen_helper_gvec_vfll64; 3173 } 3174 break; 3175 default: 3176 break; 3177 } 3178 3179 if (!fn || extract32(m4, 0, 3)) { 3180 gen_program_exception(s, PGM_SPECIFICATION); 3181 return DISAS_NORETURN; 3182 } 3183 3184 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), cpu_env, m4, fn); 3185 return DISAS_NEXT; 3186} 3187 3188static DisasJumpType op_vfmax(DisasContext *s, DisasOps *o) 3189{ 3190 const uint8_t fpf = get_field(s, m4); 3191 const uint8_t m6 = get_field(s, m6); 3192 const uint8_t m5 = get_field(s, m5); 3193 gen_helper_gvec_3_ptr *fn; 3194 3195 if (m6 == 5 || m6 == 6 || m6 == 7 || m6 > 13) { 3196 gen_program_exception(s, PGM_SPECIFICATION); 3197 return DISAS_NORETURN; 3198 } 3199 3200 switch (fpf) { 3201 case FPF_SHORT: 3202 if (s->fields.op2 == 0xef) { 3203 fn = gen_helper_gvec_vfmax32; 3204 } else { 3205 fn = gen_helper_gvec_vfmin32; 3206 } 3207 break; 3208 case FPF_LONG: 3209 if (s->fields.op2 == 0xef) { 3210 fn = gen_helper_gvec_vfmax64; 3211 } else { 3212 fn = gen_helper_gvec_vfmin64; 3213 } 3214 break; 3215 case FPF_EXT: 3216 if (s->fields.op2 == 0xef) { 3217 fn = gen_helper_gvec_vfmax128; 3218 } else { 3219 fn = gen_helper_gvec_vfmin128; 3220 } 3221 break; 3222 default: 3223 gen_program_exception(s, PGM_SPECIFICATION); 3224 return DISAS_NORETURN; 3225 } 3226 3227 gen_gvec_3_ptr(get_field(s, v1), get_field(s, v2), get_field(s, v3), 3228 cpu_env, deposit32(m5, 4, 4, m6), fn); 3229 return DISAS_NEXT; 3230} 3231 3232static DisasJumpType op_vfma(DisasContext *s, DisasOps *o) 3233{ 3234 const uint8_t m5 = get_field(s, m5); 3235 const uint8_t fpf = get_field(s, m6); 3236 gen_helper_gvec_4_ptr *fn = NULL; 3237 3238 switch (s->fields.op2) { 3239 case 0x8f: 3240 switch (fpf) { 3241 case FPF_SHORT: 3242 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3243 fn = gen_helper_gvec_vfma32; 3244 } 3245 break; 3246 case FPF_LONG: 3247 fn = gen_helper_gvec_vfma64; 3248 break; 3249 case FPF_EXT: 3250 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3251 fn = gen_helper_gvec_vfma128; 3252 } 3253 break; 3254 default: 3255 break; 3256 } 3257 break; 3258 case 0x8e: 3259 switch (fpf) { 3260 case FPF_SHORT: 3261 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3262 fn = gen_helper_gvec_vfms32; 3263 } 3264 break; 3265 case FPF_LONG: 3266 fn = gen_helper_gvec_vfms64; 3267 break; 3268 case FPF_EXT: 3269 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3270 fn = gen_helper_gvec_vfms128; 3271 } 3272 break; 3273 default: 3274 break; 3275 } 3276 break; 3277 case 0x9f: 3278 switch (fpf) { 3279 case FPF_SHORT: 3280 fn = gen_helper_gvec_vfnma32; 3281 break; 3282 case FPF_LONG: 3283 fn = gen_helper_gvec_vfnma64; 3284 break; 3285 case FPF_EXT: 3286 fn = gen_helper_gvec_vfnma128; 3287 break; 3288 default: 3289 break; 3290 } 3291 break; 3292 case 0x9e: 3293 switch (fpf) { 3294 case FPF_SHORT: 3295 fn = gen_helper_gvec_vfnms32; 3296 break; 3297 case FPF_LONG: 3298 fn = gen_helper_gvec_vfnms64; 3299 break; 3300 case FPF_EXT: 3301 fn = gen_helper_gvec_vfnms128; 3302 break; 3303 default: 3304 break; 3305 } 3306 break; 3307 default: 3308 g_assert_not_reached(); 3309 } 3310 3311 if (!fn || extract32(m5, 0, 3)) { 3312 gen_program_exception(s, PGM_SPECIFICATION); 3313 return DISAS_NORETURN; 3314 } 3315 3316 gen_gvec_4_ptr(get_field(s, v1), get_field(s, v2), 3317 get_field(s, v3), get_field(s, v4), cpu_env, m5, fn); 3318 return DISAS_NEXT; 3319} 3320 3321static DisasJumpType op_vfpso(DisasContext *s, DisasOps *o) 3322{ 3323 const uint8_t v1 = get_field(s, v1); 3324 const uint8_t v2 = get_field(s, v2); 3325 const uint8_t fpf = get_field(s, m3); 3326 const uint8_t m4 = get_field(s, m4); 3327 const uint8_t m5 = get_field(s, m5); 3328 const bool se = extract32(m4, 3, 1); 3329 TCGv_i64 tmp; 3330 3331 if ((fpf != FPF_LONG && !s390_has_feat(S390_FEAT_VECTOR_ENH)) || 3332 extract32(m4, 0, 3) || m5 > 2) { 3333 gen_program_exception(s, PGM_SPECIFICATION); 3334 return DISAS_NORETURN; 3335 } 3336 3337 switch (fpf) { 3338 case FPF_SHORT: 3339 if (!se) { 3340 switch (m5) { 3341 case 0: 3342 /* sign bit is inverted (complement) */ 3343 gen_gvec_fn_2i(xori, ES_32, v1, v2, 1ull << 31); 3344 break; 3345 case 1: 3346 /* sign bit is set to one (negative) */ 3347 gen_gvec_fn_2i(ori, ES_32, v1, v2, 1ull << 31); 3348 break; 3349 case 2: 3350 /* sign bit is set to zero (positive) */ 3351 gen_gvec_fn_2i(andi, ES_32, v1, v2, (1ull << 31) - 1); 3352 break; 3353 } 3354 return DISAS_NEXT; 3355 } 3356 break; 3357 case FPF_LONG: 3358 if (!se) { 3359 switch (m5) { 3360 case 0: 3361 /* sign bit is inverted (complement) */ 3362 gen_gvec_fn_2i(xori, ES_64, v1, v2, 1ull << 63); 3363 break; 3364 case 1: 3365 /* sign bit is set to one (negative) */ 3366 gen_gvec_fn_2i(ori, ES_64, v1, v2, 1ull << 63); 3367 break; 3368 case 2: 3369 /* sign bit is set to zero (positive) */ 3370 gen_gvec_fn_2i(andi, ES_64, v1, v2, (1ull << 63) - 1); 3371 break; 3372 } 3373 return DISAS_NEXT; 3374 } 3375 break; 3376 case FPF_EXT: 3377 /* Only a single element. */ 3378 break; 3379 default: 3380 gen_program_exception(s, PGM_SPECIFICATION); 3381 return DISAS_NORETURN; 3382 } 3383 3384 /* With a single element, we are only interested in bit 0. */ 3385 tmp = tcg_temp_new_i64(); 3386 read_vec_element_i64(tmp, v2, 0, ES_64); 3387 switch (m5) { 3388 case 0: 3389 /* sign bit is inverted (complement) */ 3390 tcg_gen_xori_i64(tmp, tmp, 1ull << 63); 3391 break; 3392 case 1: 3393 /* sign bit is set to one (negative) */ 3394 tcg_gen_ori_i64(tmp, tmp, 1ull << 63); 3395 break; 3396 case 2: 3397 /* sign bit is set to zero (positive) */ 3398 tcg_gen_andi_i64(tmp, tmp, (1ull << 63) - 1); 3399 break; 3400 } 3401 write_vec_element_i64(tmp, v1, 0, ES_64); 3402 3403 if (fpf == FPF_EXT) { 3404 read_vec_element_i64(tmp, v2, 1, ES_64); 3405 write_vec_element_i64(tmp, v1, 1, ES_64); 3406 } 3407 3408 tcg_temp_free_i64(tmp); 3409 3410 return DISAS_NEXT; 3411} 3412 3413static DisasJumpType op_vfsq(DisasContext *s, DisasOps *o) 3414{ 3415 const uint8_t fpf = get_field(s, m3); 3416 const uint8_t m4 = get_field(s, m4); 3417 gen_helper_gvec_2_ptr *fn = NULL; 3418 3419 switch (fpf) { 3420 case FPF_SHORT: 3421 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3422 fn = gen_helper_gvec_vfsq32; 3423 } 3424 break; 3425 case FPF_LONG: 3426 fn = gen_helper_gvec_vfsq64; 3427 break; 3428 case FPF_EXT: 3429 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3430 fn = gen_helper_gvec_vfsq128; 3431 } 3432 break; 3433 default: 3434 break; 3435 } 3436 3437 if (!fn || extract32(m4, 0, 3)) { 3438 gen_program_exception(s, PGM_SPECIFICATION); 3439 return DISAS_NORETURN; 3440 } 3441 3442 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), cpu_env, m4, fn); 3443 return DISAS_NEXT; 3444} 3445 3446static DisasJumpType op_vftci(DisasContext *s, DisasOps *o) 3447{ 3448 const uint16_t i3 = get_field(s, i3); 3449 const uint8_t fpf = get_field(s, m4); 3450 const uint8_t m5 = get_field(s, m5); 3451 gen_helper_gvec_2_ptr *fn = NULL; 3452 3453 switch (fpf) { 3454 case FPF_SHORT: 3455 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3456 fn = gen_helper_gvec_vftci32; 3457 } 3458 break; 3459 case FPF_LONG: 3460 fn = gen_helper_gvec_vftci64; 3461 break; 3462 case FPF_EXT: 3463 if (s390_has_feat(S390_FEAT_VECTOR_ENH)) { 3464 fn = gen_helper_gvec_vftci128; 3465 } 3466 break; 3467 default: 3468 break; 3469 } 3470 3471 if (!fn || extract32(m5, 0, 3)) { 3472 gen_program_exception(s, PGM_SPECIFICATION); 3473 return DISAS_NORETURN; 3474 } 3475 3476 gen_gvec_2_ptr(get_field(s, v1), get_field(s, v2), cpu_env, 3477 deposit32(m5, 4, 12, i3), fn); 3478 set_cc_static(s); 3479 return DISAS_NEXT; 3480} 3481