diff --git a/lib/SPIRV/SPIRVReader.cpp b/lib/SPIRV/SPIRVReader.cpp index 2542b3066..93c4a7903 100644 --- a/lib/SPIRV/SPIRVReader.cpp +++ b/lib/SPIRV/SPIRVReader.cpp @@ -1814,9 +1814,28 @@ Value *SPIRVToLLVM::transValueWithoutDecoration(SPIRVValue *BV, Function *F, return mapValue(BV, LI); } + case OpCopyMemory: { + auto *BC = static_cast(BV); + llvm::Value *Dst = transValue(BC->getTarget(), F, BB); + MaybeAlign Align(BC->getAlignment()); + MaybeAlign SrcAlign = + BC->getSrcAlignment() ? MaybeAlign(BC->getSrcAlignment()) : Align; + Type *EltTy = + transType(BC->getSource()->getType()->getPointerElementType()); + uint64_t Size = M->getDataLayout().getTypeStoreSize(EltTy).getFixedValue(); + bool IsVolatile = BC->SPIRVMemoryAccess::isVolatile(); + IRBuilder<> Builder(BB); + + llvm::Value *Src = transValue(BC->getSource(), F, BB); + CallInst *CI = + Builder.CreateMemCpy(Dst, Align, Src, SrcAlign, Size, IsVolatile); + if (isFuncNoUnwind()) + CI->getFunction()->addFnAttr(Attribute::NoUnwind); + return mapValue(BV, CI); + } + case OpCopyMemorySized: { SPIRVCopyMemorySized *BC = static_cast(BV); - CallInst *CI = nullptr; llvm::Value *Dst = transValue(BC->getTarget(), F, BB); MaybeAlign Align(BC->getAlignment()); MaybeAlign SrcAlign = @@ -1825,10 +1844,9 @@ Value *SPIRVToLLVM::transValueWithoutDecoration(SPIRVValue *BV, Function *F, bool IsVolatile = BC->SPIRVMemoryAccess::isVolatile(); IRBuilder<> Builder(BB); - if (!CI) { - llvm::Value *Src = transValue(BC->getSource(), F, BB); - CI = Builder.CreateMemCpy(Dst, Align, Src, SrcAlign, Size, IsVolatile); - } + llvm::Value *Src = transValue(BC->getSource(), F, BB); + CallInst *CI = + Builder.CreateMemCpy(Dst, Align, Src, SrcAlign, Size, IsVolatile); if (isFuncNoUnwind()) CI->getFunction()->addFnAttr(Attribute::NoUnwind); return mapValue(BV, CI); diff --git a/test/OpCopyMemory.spvasm b/test/OpCopyMemory.spvasm new file mode 100644 index 000000000..2d20981f3 --- /dev/null +++ b/test/OpCopyMemory.spvasm @@ -0,0 +1,51 @@ +; Check SPIRVReader support for OpCopyMemory. + +; REQUIRES: spirv-as +; RUN: spirv-as --target-env spv1.0 -o %t.spv %s +; RUN: spirv-val %t.spv +; RUN: llvm-spirv -r %t.spv -o %t.rev.bc +; RUN: llvm-dis %t.rev.bc -o - | FileCheck %s + + OpCapability Addresses + OpCapability Int16 + OpCapability Kernel + OpMemoryModel Physical64 OpenCL + OpEntryPoint Kernel %kernel "copymemory" + OpName %pStruct "pStruct" + OpName %dstStruct "dstStruct" + OpName %pShort "pShort" + OpName %dstShort "dstShort" + OpName %pInt "pInt" + OpName %dstInt "dstInt" + %ushort = OpTypeInt 16 0 + %uint = OpTypeInt 32 0 + %struct = OpTypeStruct %ushort %uint %ushort + %void = OpTypeVoid +%gptr_struct = OpTypePointer CrossWorkgroup %struct +%pptr_struct = OpTypePointer Function %struct + %gptr_short = OpTypePointer CrossWorkgroup %ushort + %pptr_short = OpTypePointer Function %ushort + %gptr_int = OpTypePointer CrossWorkgroup %uint + %pptr_int = OpTypePointer Function %uint + %kernel_sig = OpTypeFunction %void %gptr_short %gptr_int %gptr_struct + %ushort_42 = OpConstant %ushort 42 + %uint_4242 = OpConstant %uint 4242 +%struct_init = OpConstantComposite %struct %ushort_42 %uint_4242 %ushort_42 + %kernel = OpFunction %void None %kernel_sig + %dstShort = OpFunctionParameter %gptr_short + %dstInt = OpFunctionParameter %gptr_int + %dstStruct = OpFunctionParameter %gptr_struct + %entry = OpLabel + %pShort = OpVariable %pptr_short Function %ushort_42 + %pInt = OpVariable %pptr_int Function %uint_4242 + %pStruct = OpVariable %pptr_struct Function %struct_init + OpCopyMemory %dstShort %pShort + OpCopyMemory %dstInt %pInt + OpCopyMemory %dstStruct %pStruct + OpReturn + OpFunctionEnd + +; CHECK-LABEL: define spir_kernel void @copymemory(ptr addrspace(1) %dstShort, ptr addrspace(1) %dstInt, ptr addrspace(1) %dstStruct) +; CHECK: call void @llvm.memcpy.p1.p0.i64(ptr addrspace(1) %dstShort, ptr @pShort, i64 2, i1 false) +; CHECK: call void @llvm.memcpy.p1.p0.i64(ptr addrspace(1) %dstInt, ptr @pInt, i64 4, i1 false) +; CHECK: call void @llvm.memcpy.p1.p0.i64(ptr addrspace(1) %dstStruct, ptr @pStruct, i64 12, i1 false)