diff options
Diffstat (limited to 'llvm')
| -rw-r--r-- | llvm/lib/Target/X86/AsmParser/X86AsmParser.cpp | 42 | ||||
| -rw-r--r-- | llvm/test/MC/X86/intel-syntax.s | 9 | 
2 files changed, 36 insertions, 15 deletions
| diff --git a/llvm/lib/Target/X86/AsmParser/X86AsmParser.cpp b/llvm/lib/Target/X86/AsmParser/X86AsmParser.cpp index eb3d5e585e3..59d78461f49 100644 --- a/llvm/lib/Target/X86/AsmParser/X86AsmParser.cpp +++ b/llvm/lib/Target/X86/AsmParser/X86AsmParser.cpp @@ -936,17 +936,24 @@ X86AsmParser::CreateMemForInlineAsm(unsigned SegReg, const MCExpr *Disp,                                      unsigned Scale, SMLoc Start, SMLoc End,                                      unsigned Size, StringRef Identifier,                                      InlineAsmIdentifierInfo &Info){ -  if (isa<MCSymbolRefExpr>(Disp)) { -    // If this is not a VarDecl then assume it is a FuncDecl or some other label -    // reference.  We need an 'r' constraint here, so we need to create register -    // operand to ensure proper matching.  Just pick a GPR based on the size of -    // a pointer. -    if (!Info.IsVarDecl) { -      unsigned RegNo = -          is64BitMode() ? X86::RBX : (is32BitMode() ? X86::EBX : X86::BX); -      return X86Operand::CreateReg(RegNo, Start, End, /*AddressOf=*/true, -                                   SMLoc(), Identifier, Info.OpDecl); -    } +  // If this is not a VarDecl then assume it is a FuncDecl or some other label +  // reference.  We need an 'r' constraint here, so we need to create register +  // operand to ensure proper matching.  Just pick a GPR based on the size of +  // a pointer. +  if (isa<MCSymbolRefExpr>(Disp) && !Info.IsVarDecl) { +    unsigned RegNo = +        is64BitMode() ? X86::RBX : (is32BitMode() ? X86::EBX : X86::BX); +    return X86Operand::CreateReg(RegNo, Start, End, /*AddressOf=*/true, +                                 SMLoc(), Identifier, Info.OpDecl); +  } + +  // We either have a direct symbol reference, or an offset from a symbol.  The +  // parser always puts the symbol on the LHS, so look there for size +  // calculation purposes. +  const MCBinaryExpr *BinOp = dyn_cast<MCBinaryExpr>(Disp); +  bool IsSymRef = +      isa<MCSymbolRefExpr>(BinOp ? BinOp->getLHS() : Disp); +  if (IsSymRef) {      if (!Size) {        Size = Info.Type * 8; // Size is in terms of bits in this context.        if (Size) @@ -1154,7 +1161,7 @@ X86Operand *X86AsmParser::ParseIntelBracExpression(unsigned SegReg, SMLoc Start,    if (ParseIntelExpression(SM, End))      return 0; -  const MCExpr *Disp; +  const MCExpr *Disp = 0;    if (const MCExpr *Sym = SM.getSym()) {      // A symbolic displacement.      Disp = Sym; @@ -1162,9 +1169,14 @@ X86Operand *X86AsmParser::ParseIntelBracExpression(unsigned SegReg, SMLoc Start,        RewriteIntelBracExpression(InstInfo->AsmRewrites, SM.getSymName(),                                   ImmDisp, SM.getImm(), BracLoc, StartInBrac,                                   End); -  } else { -    // An immediate displacement only.    -    Disp = MCConstantExpr::Create(SM.getImm(), getContext()); +  } + +  if (SM.getImm() || !Disp) { +    const MCExpr *Imm = MCConstantExpr::Create(SM.getImm(), getContext()); +    if (Disp) +      Disp = MCBinaryExpr::CreateAdd(Disp, Imm, getContext()); +    else +      Disp = Imm;  // An immediate displacement only.    }    // Parse the dot operator (e.g., [ebx].foo.bar). diff --git a/llvm/test/MC/X86/intel-syntax.s b/llvm/test/MC/X86/intel-syntax.s index 0f75d5d82a5..540282a74c7 100644 --- a/llvm/test/MC/X86/intel-syntax.s +++ b/llvm/test/MC/X86/intel-syntax.s @@ -590,3 +590,12 @@ fdivr ST(1)  // CHECK: fxrstorq (%rax)  fxsave64 opaque ptr [rax]  fxrstor64 opaque ptr [rax] + +.bss +.globl _g0 +.text + +// CHECK: movq _g0, %rbx +// CHECK: movq _g0+8, %rcx +mov rbx, qword ptr [_g0] +mov rcx, qword ptr [_g0 + 8] | 

