On Thu, Mar 17, 2011 at 2:11 PM, WANG.Jiong <wong.kwongy...@gmail.com> wrote:
> define_split should be the correct way to handle this.
> You should first use define_split to break your 256bit pattern and generate
> legitimized 128bit rtl pattern sequence  for you processor
> mips_output_move should only be used to handle those legitimized one.
>
>  256 bit rtl pattern  ----> define_split
>                                                     |
>                                                    V
>                                             rtl pattern 1      --->
> mips_output_move () ->  corresponding instruciton
>                                             rtl pattern 2     ...
>                                             rtl pattern 3     ...
>
> And I am doubt abour you pattern
>
> (insn 31 30 32 3 hr-simd.c:28 (set (mem/c/i:V4DI (reg/f:DI 253
> virtual-stack-vars) [0 s+0 S32 A256])
>        (reg:V4DI 257 [ D.5235 ])) -1 (nil))
>
>
> You set the memeory from a register?
>
>
>
> On 03/17/2011 12:59 PM, Liu wrote:
>>
>> hi all
>> Our processor have a outrageous load insn, so I have to make gcc
>> support it. But when I tried some way, I failed.
>> When we suppose a load should be:
>> load_256      $z1, 16($fp)          ;load 256bits to a 256bits-wide
>> register.
>> we have to split it into:
>> load_low_128      $z1, 16($fp)          ;load 128bits to the low
>> 128bits of a 256bits-wide register.
>> load_low_128      $z2, 32($fp)          ;load 128bits to the low
>> 128bits of a 256bits-wide register.
>> combine_2_to_1      $z1, $z1, $z2, 0x20          ;combine them together.
>>
>> in mips_output_move, I can return a string such like "load_256      %0,
>> %1",
>> but I can't return "load_low_128      %0, %1\n
>>                              load_low_128      %2, %3\n
>>                              combine_2_to_1      %0, %0, %2, 0x20"
>> %3 is %1+16bytes offset, %0 and %2 are 256bits-wide registers, 0x20 is a
>> const.
>>
>> in define_insn "mov<mode>_internal", I can't using emit_insn(gen_xxx()).
>>
>> when I using emit_insn(gen_xxx()) in define_expand "mov<mode>" I get a
>> error like:
>> root@localhost:~/# mips64el-unknown-linux-gnu-gcc -S -march=xx xx-simd.c
>> xx-simd.c: In function 'test_vpaddd_u':
>> xx-simd.c:33:1: error: unrecognizable insn:
>> (insn 31 30 32 3 hr-simd.c:28 (set (mem/c/i:V4DI (reg/f:DI 253
>> virtual-stack-vars) [0 s+0 S32 A256])
>>         (reg:V4DI 257 [ D.5235 ])) -1 (nil))
>> xx-simd.c:33:1: internal compiler error: in extract_insn, at recog.c:2103
>> Please submit a full bug report,
>>
>> Please show me a path, thank you very much!
>>
>> -Liu
>>
>
>

Sorry  WANG
another toolchain take me several days.

split 256-bits move like:
(define_split
  [(set (match_operand:ZDSOTDWHB 0 "nonimmediate_operand")
        (match_operand:ZDSOTDWHB 1 "move_operand"))]
  "TARGET_1 && reload_completed"
  [(const_int 0)]
{
  mips_split_octupleword_move (operands[0], operands[1]);
  DONE;
})

mips_split_octupleword_move is:
void
mips_split_octupleword_move (rtx dest, rtx src)
{
  rtx addr, tmp_reg, imm8;
  enum machine_mode mode;
  enum rtx_code dest_code, src_code;

  dest_code = GET_CODE (dest);
  src_code = GET_CODE (src);
  mode = GET_MODE (dest);

  if (dest_code == REG && ZZ_REG_P (REGNO(dest)) && src_code == MEM)
    {
      mips_emit_move (dest, src);
      addr = plus_constant (src, 16);
      if (GET_MODE (dest) == V32QImode)
        tmp_reg = gen_reg_rtx (V32QImode);
      else if (GET_MODE (dest) == V16HImode)
        tmp_reg = gen_reg_rtx (V16HImode);
      else if (GET_MODE (dest) == V8SImode)
        tmp_reg = gen_reg_rtx (V8SImode);
      else if (GET_MODE (dest) == V4DImode)
        tmp_reg = gen_reg_rtx (V4DImode);
      else if (GET_MODE (dest) == V2TImode)
        tmp_reg = gen_reg_rtx (V2TImode);
      else if (GET_MODE (dest) == OImode)
        tmp_reg = gen_reg_rtx (OImode);
      else if (GET_MODE (dest) == V8SFmode)
        tmp_reg = gen_reg_rtx (V8SFmode);
      else if (GET_MODE (dest) == V4DFmode)
        tmp_reg = gen_reg_rtx (V4DFmode);
      else
        printf("Can't alloc Pseudo-Register for vload. \n");
      mips_emit_move (tmp_reg, addr);
      imm8 = GEN_INT(0x20);
      emit_insn (gen_vpermutqi (dest, dest, tmp_reg, imm8));
    }
  else if (src_code == REG && ZZ_REG_P (REGNO(src)) && dest_code ==MEM)
    {
      mips_emit_move (dest, src);
      addr = plus_constant (src, 16);
      if (GET_MODE (dest) == V32QImode)
        tmp_reg = gen_reg_rtx (V32QImode);
      else if (GET_MODE (dest) == V16HImode)
        tmp_reg = gen_reg_rtx (V16HImode);
      else if (GET_MODE (dest) == V8SImode)
        tmp_reg = gen_reg_rtx (V8SImode);
      else if (GET_MODE (dest) == V4DImode)
        tmp_reg = gen_reg_rtx (V4DImode);
      else if (GET_MODE (dest) == V2TImode)
        tmp_reg = gen_reg_rtx (V2TImode);
      else if (GET_MODE (dest) == OImode)
        tmp_reg = gen_reg_rtx (OImode);
      else if (GET_MODE (dest) == V8SFmode)
        tmp_reg = gen_reg_rtx (V8SFmode);
      else if (GET_MODE (dest) == V4DFmode)
        tmp_reg = gen_reg_rtx (V4DFmode);
      else
        printf("Can't alloc Pseudo-Register for vstore. \n");
      imm8 = GEN_INT(0x01);
      emit_insn (gen_vpextrv2ti (tmp_reg, dest, imm8));
      mips_emit_move (tmp_reg, addr);
    }
}

then mov<mode>:
(define_expand "mov<mode>"
  [(set (match_operand:ZDSOTDWHB 0)
        (match_operand:ZDSOTDWHB 1))]
  "TARGET_VECTORS"
{
  if (mips_legitimize_move (<MODE>mode, operands[0], operands[1]))
    DONE;
})

(define_insn "mov<mode>_internal"
  [(set (match_operand:ZDSOTDWHB 0 "nonimmediate_operand" "=m,Z,Z")
        (match_operand:ZDSOTDWHB 1 "move_operand"          "Z,m,Z"))]
  "TARGET_VECTORS"
  { return mips_output_move (operands[0], operands[1]); }
[(set_attr "move_type" "vmov")])

mips_output_move has this code:
  if (dest_code == REG && ZZ_REG_P (REGNO(dest)))
    {
      if (src_code == MEM)
        {
          return "vldql\t%0,%1";
        }
    }

  if (src_code == REG && HR_REG_P (REGNO(src)))
    {
      if (dest_code == MEM)
        {
          return "vstql\t%1,%0";
        }
    }


make and test it, get a error:

root@localhost:~/hr1-toolchain/test# mips64el-unknown-linux-gnu-gcc
-march=zz1 -S vpaddd.c
vpaddd.c: In function 'test_vpaddd_u':
vpaddd.c:33:1: internal compiler error: in gen_reg_rtx, at emit-rtl.c:863
Please submit a full bug report,
with preprocessed source if appropriate.
See <http://gcc.gnu.org/bugs.html> for instructions.

emit-rtl.c:863 is  gcc_assert (can_create_pseudo_p ());


I have no ideas...

Reply via email to