<div dir="ltr">Ah OK. No concerns then, thanks!</div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Nov 17, 2015 at 6:33 PM, Dan Gohman <span dir="ltr"><<a href="mailto:sunfish@mozilla.com" target="_blank">sunfish@mozilla.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><div>Yes, the former. Inline assembly doesn't use special syntax in the output. To make it valid, the user has to write code using the same assembler syntax as the rest of the code the compiler is emitting.<span class="HOEnZb"><font color="#888888"><br><br></font></span></div><span class="HOEnZb"><font color="#888888">Dan<br></font></span></div><div class="HOEnZb"><div class="h5"><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Nov 17, 2015 at 2:56 PM, JF Bastien <span dir="ltr"><<a href="mailto:jfb@chromium.org" target="_blank">jfb@chromium.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">I'm wondering if you see this as a pure LLVM thing that disappears in the final assembly / binary code, or if tools that consume assembly / binary code also have to understand this inline assembly. I'm totally fine with the former, and am slightly concerned by the later (it could work, but we have to make sure we spec it right). From your description it sounds like it's only the former you're going for?</div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Nov 17, 2015 at 12:11 PM, Dan Gohman <span dir="ltr"><<a href="mailto:sunfish@mozilla.com" target="_blank">sunfish@mozilla.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">asm("" : "=r" (x)) is a convenient way to force a use of x in a register while introducing minimal disruption, for the purpose of writing a compiler test. One can create a variety of contrived constraints this way.<br><br>Also, inline asm is convenient for writing tests of WebAssembly engines too, where one wants to craft a very specific code sequence in the middle of a larger program without having to write the whole function in assembly.<br><br>This just adds the "r", " m", and "i" constraints. We won't need a lot more because the ISA is very regular. Which other tools are you concerned about?<span><font color="#888888"><br><br>Dan<br></font></span></div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Nov 17, 2015 at 10:46 AM, JF Bastien <span dir="ltr"><<a href="mailto:jfb@chromium.org" target="_blank">jfb@chromium.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">Do you have concrete examples in mind? This was a tad surprising to folks here (self included), and I want to make sure that all our tools support the same feature set (i.e. we don't end up with the slightly mad situation of GCC inline asm).</div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Tue, Nov 17, 2015 at 7:09 AM, Dan Gohman <span dir="ltr"><<a href="mailto:sunfish@mozilla.com" target="_blank">sunfish@mozilla.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr"><div>We do have __builtin functions for all instructions not accessible through plain C syntax, so inline asm will almost never be needed. However, inline asm is a useful tool for writing certain kinds of testcases, both  for LLVM itself and for the underlying platform.<span><font color="#888888"><br></font></span></div><span><font color="#888888"><div><br></div>Dan<br><br></font></span></div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Mon, Nov 16, 2015 at 4:51 PM, JF Bastien <span dir="ltr"><<a href="mailto:jfb@chromium.org" target="_blank">jfb@chromium.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">(catching up on email post-vacation)<div><br></div><div>I'm a bit confused about this: the code looks like what I'd expect, but I thought we weren't going to do inline asm. Could you clarify the usecase? I thought we'd do calls to the embedders / syscalls through extern C functions, so I'm not sure I get the use for inline asm.</div></div><div><div><div class="gmail_extra"><br><div class="gmail_quote">On Thu, Nov 12, 2015 at 5:42 PM, Dan Gohman via llvm-commits <span dir="ltr"><<a href="mailto:llvm-commits@lists.llvm.org" target="_blank">llvm-commits@lists.llvm.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Author: djg<br>
Date: Thu Nov 12 19:42:29 2015<br>
New Revision: 252997<br>
<br>
URL: <a href="http://llvm.org/viewvc/llvm-project?rev=252997&view=rev" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project?rev=252997&view=rev</a><br>
Log:<br>
[WebAssembly] Inline asm support.<br>
<br>
Added:<br>
    llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll<br>
Modified:<br>
    llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp<br>
    llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp<br>
    llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp<br>
    llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h<br>
<br>
Modified: llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp?rev=252997&r1=252996&r2=252997&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp?rev=252997&r1=252996&r2=252997&view=diff</a><br>
==============================================================================<br>
--- llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp (original)<br>
+++ llvm/trunk/lib/Target/WebAssembly/WebAssemblyAsmPrinter.cpp Thu Nov 12 19:42:29 2015<br>
@@ -77,6 +77,12 @@ private:<br>
   void EmitFunctionBodyStart() override;<br>
   void EmitInstruction(const MachineInstr *MI) override;<br>
   void EmitEndOfAsmFile(Module &M) override;<br>
+  bool PrintAsmOperand(const MachineInstr *MI, unsigned OpNo,<br>
+                       unsigned AsmVariant, const char *ExtraCode,<br>
+                       raw_ostream &OS) override;<br>
+  bool PrintAsmMemoryOperand(const MachineInstr *MI, unsigned OpNo,<br>
+                             unsigned AsmVariant, const char *ExtraCode,<br>
+                             raw_ostream &OS) override;<br>
<br>
   std::string getRegTypeName(unsigned RegNo) const;<br>
   const char *toString(MVT VT) const;<br>
@@ -275,6 +281,41 @@ void WebAssemblyAsmPrinter::EmitEndOfAsm<br>
     OutStreamer->EmitRawText(Text.substr(0, Text.size() - 1));<br>
 }<br>
<br>
+bool WebAssemblyAsmPrinter::PrintAsmOperand(const MachineInstr *MI,<br>
+                                            unsigned OpNo, unsigned AsmVariant,<br>
+                                            const char *ExtraCode,<br>
+                                            raw_ostream &OS) {<br>
+  if (AsmVariant != 0)<br>
+    report_fatal_error("There are no defined alternate asm variants");<br>
+<br>
+  if (!ExtraCode) {<br>
+    const MachineOperand &MO = MI->getOperand(OpNo);<br>
+    if (MO.isImm())<br>
+      OS << MO.getImm();<br>
+    else<br>
+      OS << regToString(MO);<br>
+    return false;<br>
+  }<br>
+<br>
+  return AsmPrinter::PrintAsmOperand(MI, OpNo, AsmVariant, ExtraCode, OS);<br>
+}<br>
+<br>
+bool WebAssemblyAsmPrinter::PrintAsmMemoryOperand(const MachineInstr *MI,<br>
+                                                  unsigned OpNo,<br>
+                                                  unsigned AsmVariant,<br>
+                                                  const char *ExtraCode,<br>
+                                                  raw_ostream &OS) {<br>
+  if (AsmVariant != 0)<br>
+    report_fatal_error("There are no defined alternate asm variants");<br>
+<br>
+  if (!ExtraCode) {<br>
+    OS << regToString(MI->getOperand(OpNo));<br>
+    return false;<br>
+  }<br>
+<br>
+  return AsmPrinter::PrintAsmMemoryOperand(MI, OpNo, AsmVariant, ExtraCode, OS);<br>
+}<br>
+<br>
 // Force static initialization.<br>
 extern "C" void LLVMInitializeWebAssemblyAsmPrinter() {<br>
   RegisterAsmPrinter<WebAssemblyAsmPrinter> X(TheWebAssemblyTarget32);<br>
<br>
Modified: llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp?rev=252997&r1=252996&r2=252997&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp?rev=252997&r1=252996&r2=252997&view=diff</a><br>
==============================================================================<br>
--- llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp (original)<br>
+++ llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelDAGToDAG.cpp Thu Nov 12 19:42:29 2015<br>
@@ -56,6 +56,9 @@ public:<br>
<br>
   SDNode *Select(SDNode *Node) override;<br>
<br>
+  bool SelectInlineAsmMemoryOperand(const SDValue &Op, unsigned ConstraintID,<br>
+                                    std::vector<SDValue> &OutOps) override;<br>
+<br>
 // Include the pieces autogenerated from the target description.<br>
 #include "WebAssemblyGenDAGISel.inc"<br>
<br>
@@ -101,6 +104,22 @@ SDNode *WebAssemblyDAGToDAGISel::Select(<br>
   return ResNode;<br>
 }<br>
<br>
+bool WebAssemblyDAGToDAGISel::SelectInlineAsmMemoryOperand(<br>
+    const SDValue &Op, unsigned ConstraintID, std::vector<SDValue> &OutOps) {<br>
+  switch (ConstraintID) {<br>
+  case InlineAsm::Constraint_i:<br>
+  case InlineAsm::Constraint_m:<br>
+    // We just support simple memory operands that just have a single address<br>
+    // operand and need no special handling.<br>
+    OutOps.push_back(Op);<br>
+    return false;<br>
+  default:<br>
+    break;<br>
+  }<br>
+<br>
+  return true;<br>
+}<br>
+<br>
 /// This pass converts a legalized DAG into a WebAssembly-specific DAG, ready<br>
 /// for instruction scheduling.<br>
 FunctionPass *llvm::createWebAssemblyISelDag(WebAssemblyTargetMachine &TM,<br>
<br>
Modified: llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp?rev=252997&r1=252996&r2=252997&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp?rev=252997&r1=252996&r2=252997&view=diff</a><br>
==============================================================================<br>
--- llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp (original)<br>
+++ llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.cpp Thu Nov 12 19:42:29 2015<br>
@@ -211,6 +211,23 @@ WebAssemblyTargetLowering::getTargetNode<br>
   return nullptr;<br>
 }<br>
<br>
+std::pair<unsigned, const TargetRegisterClass *><br>
+WebAssemblyTargetLowering::getRegForInlineAsmConstraint(<br>
+    const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const {<br>
+  // First, see if this is a constraint that directly corresponds to a<br>
+  // WebAssembly register class.<br>
+  if (Constraint.size() == 1) {<br>
+    switch (Constraint[0]) {<br>
+    case 'r':<br>
+      return std::make_pair(0U, &WebAssembly::I32RegClass);<br>
+    default:<br>
+      break;<br>
+    }<br>
+  }<br>
+<br>
+  return TargetLowering::getRegForInlineAsmConstraint(TRI, Constraint, VT);<br>
+}<br>
+<br>
 //===----------------------------------------------------------------------===//<br>
 // WebAssembly Lowering private implementation.<br>
 //===----------------------------------------------------------------------===//<br>
<br>
Modified: llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h?rev=252997&r1=252996&r2=252997&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h?rev=252997&r1=252996&r2=252997&view=diff</a><br>
==============================================================================<br>
--- llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h (original)<br>
+++ llvm/trunk/lib/Target/WebAssembly/WebAssemblyISelLowering.h Thu Nov 12 19:42:29 2015<br>
@@ -49,6 +49,9 @@ private:<br>
   bool isOffsetFoldingLegal(const GlobalAddressSDNode *GA) const override;<br>
   MVT getScalarShiftAmountTy(const DataLayout &DL, EVT) const override;<br>
   const char *getTargetNodeName(unsigned Opcode) const override;<br>
+  std::pair<unsigned, const TargetRegisterClass *><br>
+  getRegForInlineAsmConstraint(const TargetRegisterInfo *TRI,<br>
+                               StringRef Constraint, MVT VT) const override;<br>
<br>
   SDValue LowerCall(CallLoweringInfo &CLI,<br>
                     SmallVectorImpl<SDValue> &InVals) const override;<br>
<br>
Added: llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll?rev=252997&view=auto" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll?rev=252997&view=auto</a><br>
==============================================================================<br>
--- llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll (added)<br>
+++ llvm/trunk/test/CodeGen/WebAssembly/inline-asm.ll Thu Nov 12 19:42:29 2015<br>
@@ -0,0 +1,53 @@<br>
+; RUN: llc < %s -asm-verbose=false | FileCheck %s<br>
+<br>
+; Test basic inline assembly.<br>
+<br>
+target datalayout = "e-p:32:32-i64:64-n32:64-S128"<br>
+target triple = "wasm32-unknown-unknown"<br>
+<br>
+; CHECK-LABEL: foo:<br>
+; CHECK-NEXT: .param i32{{$}}<br>
+; CHECK-NEXT: .result i32{{$}}<br>
+; CHECK-NEXT: .local i32, i32{{$}}<br>
+; CHECK-NEXT: #APP{{$}}<br>
+; CHECK-NEXT: 1 = aaa(0){{$}}<br>
+; CHECK-NEXT: #NO_APP{{$}}<br>
+; CHECK-NEXT: return (get_local 1){{$}}<br>
+define i32 @foo(i32 %r) {<br>
+entry:<br>
+  %0 = tail call i32 asm sideeffect "$0 = aaa($1)", "=r,r"(i32 %r) #0, !srcloc !0<br>
+  ret i32 %0<br>
+}<br>
+<br>
+; CHECK-LABEL: bar:<br>
+; CHECK-NEXT: .param i32{{$}}<br>
+; CHECK-NEXT: .param i32{{$}}<br>
+; CHECK-NEXT: .local i32, i32{{$}}<br>
+; CHECK-NEXT: #APP{{$}}<br>
+; CHECK-NEXT: 1 = bbb(0){{$}}<br>
+; CHECK-NEXT: #NO_APP{{$}}<br>
+; CHECK-NEXT: return{{$}}<br>
+define void @bar(i32* %r, i32* %s) {<br>
+entry:<br>
+  tail call void asm sideeffect "$0 = bbb($1)", "=*m,*m"(i32* %s, i32* %r) #0, !srcloc !1<br>
+  ret void<br>
+}<br>
+<br>
+; CHECK-LABEL: imm:<br>
+; CHECK-NEXT: .result i32{{$}}<br>
+; CHECK-NEXT: .local i32{{$}}<br>
+; CHECK-NEXT: #APP{{$}}<br>
+; CHECK-NEXT: 0 = ccc(42){{$}}<br>
+; CHECK-NEXT: #NO_APP{{$}}<br>
+; CHECK-NEXT: return (get_local 0){{$}}<br>
+define i32 @imm() {<br>
+entry:<br>
+  %0 = tail call i32 asm sideeffect "$0 = ccc($1)", "=r,i"(i32 42) #0, !srcloc !2<br>
+  ret i32 %0<br>
+}<br>
+<br>
+attributes #0 = { nounwind }<br>
+<br>
+!0 = !{i32 47}<br>
+!1 = !{i32 145}<br>
+!2 = !{i32 231}<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><br></div>
</div></div></blockquote></div><br></div>
</div></div></blockquote></div><br></div>
</div></div></blockquote></div><br></div>
</div></div></blockquote></div><br></div>
</div></div></blockquote></div><br></div>
</div></div></blockquote></div><br></div>