<p dir="ltr">Sure. Thanks for letting me know about the changes you want. </p>
<br><div class="gmail_quote"><div dir="ltr">On Tue 9 May, 2017, 16:54 Michael Kruse, <<a href="mailto:llvm-commits@meinersbur.de">llvm-commits@meinersbur.de</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Can you remove the "[Polly]" prefix before committing? It's a commit<br>
to the Polly repository, so it does not add new information. The<br>
distinction of repositories does not exist in Phabricator, which is<br>
the reason for the prefix there.<br>
<br>
When I do commits, I usually also do some normalization, but I don't<br>
personally care:<br>
- Put a dot at the end of the title.<br>
- Remove "Summary:"<br>
- Wrap after 72 characters<br>
- Keep only the "Differential Revision" line. The other information<br>
about the review process is already available behind the link.<br>
<br>
This is what I found in most LLVM commits.<br>
<br>
Thank you.<br>
<br>
Michael<br>
<br>
2017-05-09 12:45 GMT+02:00 Siddharth Bhat via llvm-commits<br>
<<a href="mailto:llvm-commits@lists.llvm.org" target="_blank">llvm-commits@lists.llvm.org</a>>:<br>
> Author: bollu<br>
> Date: Tue May  9 05:45:52 2017<br>
> New Revision: 302515<br>
><br>
> URL: <a href="http://llvm.org/viewvc/llvm-project?rev=302515&view=rev" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project?rev=302515&view=rev</a><br>
> Log:<br>
> [Polly][PPCGCodeGen] OpenCL now gets kernel argument size from PPCG CodeGen<br>
><br>
> Summary: PPCGCodeGeneration now attaches the size of the kernel launch parameters at the end of the parameter list. For the existing CUDA Runtime, this gets ignored, but the OpenCL Runtime knows to check for kernel-argument size at the end of the parameter list. (The resulting parameters list is twice as long. This has been accounted for in the corresponding test cases).<br>
><br>
> Reviewers: grosser, Meinersbur, bollu<br>
><br>
> Reviewed By: bollu<br>
><br>
> Subscribers: nemanjai, yaxunl, Anastasia, pollydev, llvm-commits<br>
><br>
> Tags: #polly<br>
><br>
> Differential Revision: <a href="https://reviews.llvm.org/D32961" rel="noreferrer" target="_blank">https://reviews.llvm.org/D32961</a><br>
><br>
> Modified:<br>
>     polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp<br>
>     polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll<br>
>     polly/trunk/test/GPGPU/host-control-flow.ll<br>
>     polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll<br>
>     polly/trunk/test/GPGPU/parametric-loop-bound.ll<br>
>     polly/trunk/tools/GPURuntime/GPUJIT.c<br>
><br>
> Modified: polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp (original)<br>
> +++ polly/trunk/lib/CodeGen/PPCGCodeGeneration.cpp Tue May  9 05:45:52 2017<br>
> @@ -142,6 +142,14 @@ static __isl_give isl_id_to_ast_expr *po<br>
>    return RefToExpr;<br>
>  }<br>
><br>
> +/// Given a LLVM Type, compute its size in bytes,<br>
> +static int computeSizeInBytes(const Type *T) {<br>
> +  int bytes = T->getPrimitiveSizeInBits() / 8;<br>
> +  if (bytes == 0)<br>
> +    bytes = T->getScalarSizeInBits() / 8;<br>
> +  return bytes;<br>
> +}<br>
> +<br>
>  /// Generate code for a GPU specific isl AST.<br>
>  ///<br>
>  /// The GPUNodeBuilder augments the general existing IslNodeBuilder, which<br>
> @@ -272,6 +280,16 @@ private:<br>
>    /// @returns A tuple with thread block sizes for X, Y, and Z dimensions.<br>
>    std::tuple<Value *, Value *, Value *> getBlockSizes(ppcg_kernel *Kernel);<br>
><br>
> +  /// Store a specific kernel launch parameter in the array of kernel launch<br>
> +  /// parameters.<br>
> +  ///<br>
> +  /// @param Parameters The list of parameters in which to store.<br>
> +  /// @param Param      The kernel launch parameter to store.<br>
> +  /// @param Index      The index in the parameter list, at which to store the<br>
> +  ///                   parameter.<br>
> +  void insertStoreParameter(Instruction *Parameters, Instruction *Param,<br>
> +                            int Index);<br>
> +<br>
>    /// Create kernel launch parameters.<br>
>    ///<br>
>    /// @param Kernel        The kernel to create parameters for.<br>
> @@ -1192,11 +1210,21 @@ GPUNodeBuilder::getBlockSizes(ppcg_kerne<br>
>    return std::make_tuple(Sizes[0], Sizes[1], Sizes[2]);<br>
>  }<br>
><br>
> +void GPUNodeBuilder::insertStoreParameter(Instruction *Parameters,<br>
> +                                          Instruction *Param, int Index) {<br>
> +  Value *Slot = Builder.CreateGEP(<br>
> +      Parameters, {Builder.getInt64(0), Builder.getInt64(Index)});<br>
> +  Value *ParamTyped = Builder.CreatePointerCast(Param, Builder.getInt8PtrTy());<br>
> +  Builder.CreateStore(ParamTyped, Slot);<br>
> +}<br>
> +<br>
>  Value *<br>
>  GPUNodeBuilder::createLaunchParameters(ppcg_kernel *Kernel, Function *F,<br>
>                                         SetVector<Value *> SubtreeValues) {<br>
> -  Type *ArrayTy = ArrayType::get(Builder.getInt8PtrTy(),<br>
> -                                 std::distance(F->arg_begin(), F->arg_end()));<br>
> +  const int NumArgs = F->arg_size();<br>
> +  std::vector<int> ArgSizes(NumArgs);<br>
> +<br>
> +  Type *ArrayTy = ArrayType::get(Builder.getInt8PtrTy(), 2 * NumArgs);<br>
><br>
>    BasicBlock *EntryBlock =<br>
>        &Builder.GetInsertBlock()->getParent()->getEntryBlock();<br>
> @@ -1213,6 +1241,8 @@ GPUNodeBuilder::createLaunchParameters(p<br>
>      isl_id *Id = isl_space_get_tuple_id(Prog->array[i].space, isl_dim_set);<br>
>      const ScopArrayInfo *SAI = ScopArrayInfo::getFromId(Id);<br>
><br>
> +    ArgSizes[Index] = SAI->getElemSizeInBytes();<br>
> +<br>
>      Value *DevArray = nullptr;<br>
>      if (ManagedMemory) {<br>
>        DevArray = getOrCreateManagedDeviceArray(<br>
> @@ -1265,16 +1295,15 @@ GPUNodeBuilder::createLaunchParameters(p<br>
>      isl_id *Id = isl_space_get_dim_id(Kernel->space, isl_dim_set, i);<br>
>      Value *Val = IDToValue[Id];<br>
>      isl_id_free(Id);<br>
> +<br>
> +    ArgSizes[Index] = computeSizeInBytes(Val->getType());<br>
> +<br>
>      Instruction *Param =<br>
>          new AllocaInst(Val->getType(), AddressSpace,<br>
>                         Launch + "_param_" + std::to_string(Index),<br>
>                         EntryBlock->getTerminator());<br>
>      Builder.CreateStore(Val, Param);<br>
> -    Value *Slot = Builder.CreateGEP(<br>
> -        Parameters, {Builder.getInt64(0), Builder.getInt64(Index)});<br>
> -    Value *ParamTyped =<br>
> -        Builder.CreatePointerCast(Param, Builder.getInt8PtrTy());<br>
> -    Builder.CreateStore(ParamTyped, Slot);<br>
> +    insertStoreParameter(Parameters, Param, Index);<br>
>      Index++;<br>
>    }<br>
><br>
> @@ -1284,30 +1313,38 @@ GPUNodeBuilder::createLaunchParameters(p<br>
>      isl_id *Id = isl_space_get_dim_id(Kernel->space, isl_dim_param, i);<br>
>      Value *Val = IDToValue[Id];<br>
>      isl_id_free(Id);<br>
> +<br>
> +    ArgSizes[Index] = computeSizeInBytes(Val->getType());<br>
> +<br>
>      Instruction *Param =<br>
>          new AllocaInst(Val->getType(), AddressSpace,<br>
>                         Launch + "_param_" + std::to_string(Index),<br>
>                         EntryBlock->getTerminator());<br>
>      Builder.CreateStore(Val, Param);<br>
> -    Value *Slot = Builder.CreateGEP(<br>
> -        Parameters, {Builder.getInt64(0), Builder.getInt64(Index)});<br>
> -    Value *ParamTyped =<br>
> -        Builder.CreatePointerCast(Param, Builder.getInt8PtrTy());<br>
> -    Builder.CreateStore(ParamTyped, Slot);<br>
> +    insertStoreParameter(Parameters, Param, Index);<br>
>      Index++;<br>
>    }<br>
><br>
>    for (auto Val : SubtreeValues) {<br>
> +    ArgSizes[Index] = computeSizeInBytes(Val->getType());<br>
> +<br>
>      Instruction *Param =<br>
>          new AllocaInst(Val->getType(), AddressSpace,<br>
>                         Launch + "_param_" + std::to_string(Index),<br>
>                         EntryBlock->getTerminator());<br>
>      Builder.CreateStore(Val, Param);<br>
> -    Value *Slot = Builder.CreateGEP(<br>
> -        Parameters, {Builder.getInt64(0), Builder.getInt64(Index)});<br>
> -    Value *ParamTyped =<br>
> -        Builder.CreatePointerCast(Param, Builder.getInt8PtrTy());<br>
> -    Builder.CreateStore(ParamTyped, Slot);<br>
> +    insertStoreParameter(Parameters, Param, Index);<br>
> +    Index++;<br>
> +  }<br>
> +<br>
> +  for (int i = 0; i < NumArgs; i++) {<br>
> +    Value *Val = ConstantInt::get(Builder.getInt32Ty(), ArgSizes[i]);<br>
> +    Instruction *Param =<br>
> +        new AllocaInst(Builder.getInt32Ty(), AddressSpace,<br>
> +                       Launch + "_param_size_" + std::to_string(i),<br>
> +                       EntryBlock->getTerminator());<br>
> +    Builder.CreateStore(Val, Param);<br>
> +    insertStoreParameter(Parameters, Param, Index);<br>
>      Index++;<br>
>    }<br>
><br>
><br>
> Modified: polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll (original)<br>
> +++ polly/trunk/test/GPGPU/cuda-managed-memory-simple.ll Tue May  9 05:45:52 2017<br>
> @@ -37,18 +37,26 @@<br>
><br>
>  ; CHECK:       %13 = call i8* @polly_initContextCUDA()<br>
>  ; CHECK-NEXT:  %14 = bitcast i32* %A to i8*<br>
> -; CHECK-NEXT:  %15 = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_0_params, i64 0, i64 0<br>
> +; CHECK-NEXT:  %15 = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 0<br>
>  ; CHECK-NEXT:  store i8* %14, i8** %polly_launch_0_param_0<br>
>  ; CHECK-NEXT:  %16 = bitcast i8** %polly_launch_0_param_0 to i8*<br>
>  ; CHECK-NEXT:  store i8* %16, i8** %15<br>
>  ; CHECK-NEXT:  %17 = bitcast i32* %R to i8*<br>
> -; CHECK-NEXT:  %18 = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
> +; CHECK-NEXT:  %18 = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
>  ; CHECK-NEXT:  store i8* %17, i8** %polly_launch_0_param_1<br>
>  ; CHECK-NEXT:  %19 = bitcast i8** %polly_launch_0_param_1 to i8*<br>
>  ; CHECK-NEXT:  store i8* %19, i8** %18<br>
> -; CHECK-NEXT:  %20 = call i8* @polly_getKernel(i8* getelementptr inbounds ([750 x i8], [750 x i8]* @kernel_0, i32 0, i32 0), i8* getelementptr inbounds ([9 x i8], [9 x i8]* @kernel_0_name, i32 0, i32 0))<br>
> -; CHECK-NEXT:  call void @polly_launchKernel(i8* %20, i32 2, i32 1, i32 32, i32 1, i32 1, i8* %polly_launch_0_params_i8ptr)<br>
> -; CHECK-NEXT:  call void @polly_freeKernel(i8* %20)<br>
> +; CHECK-NEXT:  store i32 4, i32* %polly_launch_0_param_size_0<br>
> +; CHECK-NEXT:  %20 = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 2<br>
> +; CHECK-NEXT:  %21 = bitcast i32* %polly_launch_0_param_size_0 to i8*<br>
> +; CHECK-NEXT:  store i8* %21, i8** %20<br>
> +; CHECK-NEXT:  store i32 4, i32* %polly_launch_0_param_size_1<br>
> +; CHECK-NEXT:  %22 = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 3<br>
> +; CHECK-NEXT:  %23 = bitcast i32* %polly_launch_0_param_size_1 to i8*<br>
> +; CHECK-NEXT:  store i8* %23, i8** %22<br>
> +; CHECK-NEXT:  %24 = call i8* @polly_getKernel(i8* getelementptr inbounds ([750 x i8], [750 x i8]* @kernel_0, i32 0, i32 0), i8* getelementptr inbounds ([9 x i8], [9 x i8]* @kernel_0_name, i32 0, i32 0))<br>
> +; CHECK-NEXT:  call void @polly_launchKernel(i8* %24, i32 2, i32 1, i32 32, i32 1, i32 1, i8* %polly_launch_0_params_i8ptr)<br>
> +; CHECK-NEXT:  call void @polly_freeKernel(i8* %24)<br>
>  ; CHECK-NEXT:  call void @polly_synchronizeDevice()<br>
>  ; CHECK-NEXT:  call void @polly_freeContext(i8* %13)<br>
><br>
><br>
> Modified: polly/trunk/test/GPGPU/host-control-flow.ll<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/host-control-flow.ll?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/host-control-flow.ll?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/test/GPGPU/host-control-flow.ll (original)<br>
> +++ polly/trunk/test/GPGPU/host-control-flow.ll Tue May  9 05:45:52 2017<br>
> @@ -32,7 +32,7 @@<br>
>  ; IR-NEXT:   %polly.indvar = phi i64 [ 0, %polly.loop_preheader ], [ %polly.indvar_next, %polly.loop_header ]<br>
>  ; ...<br>
>  ; IR:  store i64 %polly.indvar, i64* %polly_launch_0_param_1<br>
> -; IR-NEXT:  [[REGA:%.+]] = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
> +; IR-NEXT:  [[REGA:%.+]] = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
>  ; IR-NEXT:  [[REGB:%.+]] = bitcast i64* %polly_launch_0_param_1 to i8*<br>
>  ; IR-NEXT:  store i8* [[REGB]], i8** [[REGA]]<br>
>  ; IR: call i8* @polly_getKernel<br>
><br>
> Modified: polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll (original)<br>
> +++ polly/trunk/test/GPGPU/kernel-params-only-some-arrays.ll Tue May  9 05:45:52 2017<br>
> @@ -48,13 +48,13 @@<br>
><br>
><br>
>  ; IR:       [[DEVPTR:%.*]] = call i8* @polly_getDevicePtr(i8* %p_dev_array_MemRef_A)<br>
> -; IR-NEXT:  [[SLOT:%.*]] = getelementptr [1 x i8*], [1 x i8*]* %polly_launch_0_params, i64 0, i64 0<br>
> +; IR-NEXT:  [[SLOT:%.*]] = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_0_params, i64 0, i64 0<br>
>  ; IR-NEXT:  store i8* [[DEVPTR]], i8** %polly_launch_0_param_0<br>
>  ; IR-NEXT:  [[DATA:%.*]] = bitcast i8** %polly_launch_0_param_0 to i8*<br>
>  ; IR-NEXT:  store i8* [[DATA]], i8** [[SLOT]]<br>
><br>
>  ; IR:       [[DEVPTR:%.*]] = call i8* @polly_getDevicePtr(i8* %p_dev_array_MemRef_B)<br>
> -; IR-NEXT:  [[SLOT:%.*]] = getelementptr [1 x i8*], [1 x i8*]* %polly_launch_1_params, i64 0, i64 0<br>
> +; IR-NEXT:  [[SLOT:%.*]] = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_1_params, i64 0, i64 0<br>
>  ; IR-NEXT:  store i8* [[DEVPTR]], i8** %polly_launch_1_param_0<br>
>  ; IR-NEXT:  [[DATA:%.*]] = bitcast i8** %polly_launch_1_param_0 to i8*<br>
>  ; IR-NEXT:  store i8* [[DATA]], i8** [[SLOT]]<br>
><br>
> Modified: polly/trunk/test/GPGPU/parametric-loop-bound.ll<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/parametric-loop-bound.ll?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/test/GPGPU/parametric-loop-bound.ll?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/test/GPGPU/parametric-loop-bound.ll (original)<br>
> +++ polly/trunk/test/GPGPU/parametric-loop-bound.ll Tue May  9 05:45:52 2017<br>
> @@ -31,7 +31,7 @@<br>
>  ; CODE-NEXT:     Stmt_bb2(32 * b0 + t0 + 1048576 * c0);<br>
><br>
>  ; IR: store i64 %n, i64* %polly_launch_0_param_1<br>
> -; IR-NEXT: [[REGA:%.+]] = getelementptr [2 x i8*], [2 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
> +; IR-NEXT: [[REGA:%.+]] = getelementptr [4 x i8*], [4 x i8*]* %polly_launch_0_params, i64 0, i64 1<br>
>  ; IR-NEXT: [[REGB:%.+]] = bitcast i64* %polly_launch_0_param_1 to i8*<br>
>  ; IR-NEXT: store i8* [[REGB]], i8** [[REGA]]<br>
><br>
><br>
> Modified: polly/trunk/tools/GPURuntime/GPUJIT.c<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/polly/trunk/tools/GPURuntime/GPUJIT.c?rev=302515&r1=302514&r2=302515&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/polly/trunk/tools/GPURuntime/GPUJIT.c?rev=302515&r1=302514&r2=302515&view=diff</a><br>
> ==============================================================================<br>
> --- polly/trunk/tools/GPURuntime/GPUJIT.c (original)<br>
> +++ polly/trunk/tools/GPURuntime/GPUJIT.c Tue May  9 05:45:52 2017<br>
> @@ -554,28 +554,12 @@ static void launchKernelCL(PollyGPUFunct<br>
>                                sizeof(cl_uint), &NumArgs, NULL);<br>
>    checkOpenCLError(Ret, "Failed to get number of kernel arguments.\n");<br>
><br>
> -  // TODO: Pass the size of the kernel arguments in to launchKernelCL, along<br>
> -  // with the arguments themselves. This is a dirty workaround that can be<br>
> -  // broken.<br>
> +  /* Argument sizes are stored at the end of the Parameters array. */<br>
>    for (cl_uint i = 0; i < NumArgs; i++) {<br>
> -    Ret = clSetKernelArgFcnPtr(CLKernel->Kernel, i, 8, (void *)Parameters[i]);<br>
> -    if (Ret == CL_INVALID_ARG_SIZE) {<br>
> -      Ret = clSetKernelArgFcnPtr(CLKernel->Kernel, i, 4, (void *)Parameters[i]);<br>
> -      if (Ret == CL_INVALID_ARG_SIZE) {<br>
> -        Ret =<br>
> -            clSetKernelArgFcnPtr(CLKernel->Kernel, i, 2, (void *)Parameters[i]);<br>
> -        if (Ret == CL_INVALID_ARG_SIZE) {<br>
> -          Ret = clSetKernelArgFcnPtr(CLKernel->Kernel, i, 1,<br>
> -                                     (void *)Parameters[i]);<br>
> -          checkOpenCLError(Ret, "Failed to set Kernel argument %d.\n", i);<br>
> -        }<br>
> -      }<br>
> -    }<br>
> -    if (Ret != CL_SUCCESS && Ret != CL_INVALID_ARG_SIZE) {<br>
> -      fprintf(stderr, "Failed to set Kernel argument.\n");<br>
> -      printOpenCLError(Ret);<br>
> -      exit(-1);<br>
> -    }<br>
> +    Ret = clSetKernelArgFcnPtr(CLKernel->Kernel, i,<br>
> +                               *((int *)Parameters[NumArgs + i]),<br>
> +                               (void *)Parameters[i]);<br>
> +    checkOpenCLError(Ret, "Failed to set Kernel argument %d.\n", i);<br>
>    }<br>
><br>
>    unsigned int GridDimZ = 1;<br>
><br>
><br>
> _______________________________________________<br>
> llvm-commits mailing list<br>
> <a href="mailto:llvm-commits@lists.llvm.org" target="_blank">llvm-commits@lists.llvm.org</a><br>
> <a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-commits" rel="noreferrer" target="_blank">http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-commits</a><br>
</blockquote></div><div dir="ltr">-- <br></div><div data-smartmail="gmail_signature"><div dir="ltr">Sending this from my phone, please excuse any typos!</div></div>