[ms-inline asm] Add support for calling functions from inline assembly.

Part of rdar://12991541

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@172121 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Chad Rosier 2013-01-10 22:10:27 +00:00
parent 48fdf9b379
commit c1ec207b61
5 changed files with 66 additions and 18 deletions

View File

@ -36,7 +36,7 @@ class MCAsmParserSemaCallback {
public: public:
virtual ~MCAsmParserSemaCallback(); virtual ~MCAsmParserSemaCallback();
virtual void *LookupInlineAsmIdentifier(StringRef Name, void *Loc, virtual void *LookupInlineAsmIdentifier(StringRef Name, void *Loc,
unsigned &Size) = 0; unsigned &Size, bool &IsVarDecl) = 0;
virtual bool LookupInlineAsmField(StringRef Base, StringRef Member, virtual bool LookupInlineAsmField(StringRef Base, StringRef Member,
unsigned &Offset) = 0; unsigned &Offset) = 0;
}; };

View File

@ -70,6 +70,10 @@ public:
/// care of the rewrites. Only valid when parsing MS-style inline assembly. /// care of the rewrites. Only valid when parsing MS-style inline assembly.
virtual bool needAsmRewrite() const { return true; } virtual bool needAsmRewrite() const { return true; }
/// needAddressOf - Do we need to emit code to get the address of the
/// variable/label? Only valid when parsing MS-style inline assembly.
virtual bool needAddressOf() const { return false; }
/// isOffsetOf - Do we need to emit code to get the offset of the variable, /// isOffsetOf - Do we need to emit code to get the offset of the variable,
/// rather then the value of the variable? Only valid when parsing MS-style /// rather then the value of the variable? Only valid when parsing MS-style
/// inline assembly. /// inline assembly.

View File

@ -3777,8 +3777,8 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
MCAsmParserSemaCallback &SI) { MCAsmParserSemaCallback &SI) {
SmallVector<void *, 4> InputDecls; SmallVector<void *, 4> InputDecls;
SmallVector<void *, 4> OutputDecls; SmallVector<void *, 4> OutputDecls;
SmallVector<bool, 4> InputDeclsOffsetOf; SmallVector<bool, 4> InputDeclsAddressOf;
SmallVector<bool, 4> OutputDeclsOffsetOf; SmallVector<bool, 4> OutputDeclsAddressOf;
SmallVector<std::string, 4> InputConstraints; SmallVector<std::string, 4> InputConstraints;
SmallVector<std::string, 4> OutputConstraints; SmallVector<std::string, 4> OutputConstraints;
std::set<std::string> ClobberRegs; std::set<std::string> ClobberRegs;
@ -3815,7 +3815,7 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
} }
// Register operand. // Register operand.
if (Operand->isReg() && !Operand->isOffsetOf()) { if (Operand->isReg() && !Operand->needAddressOf()) {
unsigned NumDefs = Desc.getNumDefs(); unsigned NumDefs = Desc.getNumDefs();
// Clobber. // Clobber.
if (NumDefs && Operand->getMCOperandNum() < NumDefs) { if (NumDefs && Operand->getMCOperandNum() < NumDefs) {
@ -3829,11 +3829,12 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
// Expr/Input or Output. // Expr/Input or Output.
unsigned Size; unsigned Size;
bool IsVarDecl;
void *OpDecl = SI.LookupInlineAsmIdentifier(Operand->getName(), AsmLoc, void *OpDecl = SI.LookupInlineAsmIdentifier(Operand->getName(), AsmLoc,
Size); Size, IsVarDecl);
if (OpDecl) { if (OpDecl) {
bool isOutput = (i == 1) && Desc.mayStore(); bool isOutput = (i == 1) && Desc.mayStore();
if (!Operand->isOffsetOf() && Operand->needSizeDirective()) if (Operand->isMem() && Operand->needSizeDirective())
AsmStrRewrites.push_back(AsmRewrite(AOK_SizeDirective, AsmStrRewrites.push_back(AsmRewrite(AOK_SizeDirective,
Operand->getStartLoc(), Operand->getStartLoc(),
/*Len*/0, /*Len*/0,
@ -3842,7 +3843,7 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
std::string Constraint = "="; std::string Constraint = "=";
++InputIdx; ++InputIdx;
OutputDecls.push_back(OpDecl); OutputDecls.push_back(OpDecl);
OutputDeclsOffsetOf.push_back(Operand->isOffsetOf()); OutputDeclsAddressOf.push_back(Operand->needAddressOf());
Constraint += Operand->getConstraint().str(); Constraint += Operand->getConstraint().str();
OutputConstraints.push_back(Constraint); OutputConstraints.push_back(Constraint);
AsmStrRewrites.push_back(AsmRewrite(AOK_Output, AsmStrRewrites.push_back(AsmRewrite(AOK_Output,
@ -3850,7 +3851,7 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
Operand->getNameLen())); Operand->getNameLen()));
} else { } else {
InputDecls.push_back(OpDecl); InputDecls.push_back(OpDecl);
InputDeclsOffsetOf.push_back(Operand->isOffsetOf()); InputDeclsAddressOf.push_back(Operand->needAddressOf());
InputConstraints.push_back(Operand->getConstraint().str()); InputConstraints.push_back(Operand->getConstraint().str());
AsmStrRewrites.push_back(AsmRewrite(AOK_Input, AsmStrRewrites.push_back(AsmRewrite(AOK_Input,
Operand->getStartLoc(), Operand->getStartLoc(),
@ -3876,14 +3877,14 @@ bool AsmParser::ParseMSInlineAsm(void *AsmLoc, std::string &AsmString,
OpDecls.resize(NumExprs); OpDecls.resize(NumExprs);
Constraints.resize(NumExprs); Constraints.resize(NumExprs);
// FIXME: Constraints are hard coded to 'm', but we need an 'r' // FIXME: Constraints are hard coded to 'm', but we need an 'r'
// constraint for offsetof. This needs to be cleaned up! // constraint for addressof. This needs to be cleaned up!
for (unsigned i = 0; i < NumOutputs; ++i) { for (unsigned i = 0; i < NumOutputs; ++i) {
OpDecls[i] = std::make_pair(OutputDecls[i], OutputDeclsOffsetOf[i]); OpDecls[i] = std::make_pair(OutputDecls[i], OutputDeclsAddressOf[i]);
Constraints[i] = OutputDeclsOffsetOf[i] ? "=r" : OutputConstraints[i]; Constraints[i] = OutputDeclsAddressOf[i] ? "=r" : OutputConstraints[i];
} }
for (unsigned i = 0, j = NumOutputs; i < NumInputs; ++i, ++j) { for (unsigned i = 0, j = NumOutputs; i < NumInputs; ++i, ++j) {
OpDecls[j] = std::make_pair(InputDecls[i], InputDeclsOffsetOf[i]); OpDecls[j] = std::make_pair(InputDecls[i], InputDeclsAddressOf[i]);
Constraints[j] = InputDeclsOffsetOf[i] ? "r" : InputConstraints[i]; Constraints[j] = InputDeclsAddressOf[i] ? "r" : InputConstraints[i];
} }
} }

View File

@ -168,6 +168,7 @@ struct X86Operand : public MCParsedAsmOperand {
SMLoc StartLoc, EndLoc; SMLoc StartLoc, EndLoc;
SMLoc OffsetOfLoc; SMLoc OffsetOfLoc;
bool AddressOf;
union { union {
struct { struct {
@ -340,6 +341,10 @@ struct X86Operand : public MCParsedAsmOperand {
return OffsetOfLoc.getPointer(); return OffsetOfLoc.getPointer();
} }
bool needAddressOf() const {
return AddressOf;
}
bool needSizeDirective() const { bool needSizeDirective() const {
assert(Kind == Memory && "Invalid access!"); assert(Kind == Memory && "Invalid access!");
return Mem.NeedSizeDir; return Mem.NeedSizeDir;
@ -471,9 +476,11 @@ struct X86Operand : public MCParsedAsmOperand {
} }
static X86Operand *CreateReg(unsigned RegNo, SMLoc StartLoc, SMLoc EndLoc, static X86Operand *CreateReg(unsigned RegNo, SMLoc StartLoc, SMLoc EndLoc,
bool AddressOf = false,
SMLoc OffsetOfLoc = SMLoc()) { SMLoc OffsetOfLoc = SMLoc()) {
X86Operand *Res = new X86Operand(Register, StartLoc, EndLoc); X86Operand *Res = new X86Operand(Register, StartLoc, EndLoc);
Res->Reg.RegNo = RegNo; Res->Reg.RegNo = RegNo;
Res->AddressOf = AddressOf;
Res->OffsetOfLoc = OffsetOfLoc; Res->OffsetOfLoc = OffsetOfLoc;
return Res; return Res;
} }
@ -836,25 +843,40 @@ X86Operand *X86AsmParser::ParseIntelMemOperand(unsigned SegReg, SMLoc Start) {
return 0; return 0;
bool NeedSizeDir = false; bool NeedSizeDir = false;
if (!Size && isParsingInlineAsm()) { bool IsVarDecl = false;
if (isParsingInlineAsm()) {
if (const MCSymbolRefExpr *SymRef = dyn_cast<MCSymbolRefExpr>(Disp)) { if (const MCSymbolRefExpr *SymRef = dyn_cast<MCSymbolRefExpr>(Disp)) {
const MCSymbol &Sym = SymRef->getSymbol(); const MCSymbol &Sym = SymRef->getSymbol();
// FIXME: The SemaLookup will fail if the name is anything other then an // FIXME: The SemaLookup will fail if the name is anything other then an
// identifier. // identifier.
// FIXME: Pass a valid SMLoc. // FIXME: Pass a valid SMLoc.
SemaCallback->LookupInlineAsmIdentifier(Sym.getName(), NULL, Size); unsigned tSize;
SemaCallback->LookupInlineAsmIdentifier(Sym.getName(), NULL, tSize,
IsVarDecl);
if (!Size)
Size = tSize;
NeedSizeDir = Size > 0; NeedSizeDir = Size > 0;
} }
} }
if (!isParsingInlineAsm()) if (!isParsingInlineAsm())
return X86Operand::CreateMem(Disp, Start, End, Size); return X86Operand::CreateMem(Disp, Start, End, Size);
else else {
// 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 (!IsVarDecl) {
unsigned RegNo = is64BitMode() ? X86::RBX : X86::EBX;
return X86Operand::CreateReg(RegNo, Start, End, /*AddressOf=*/true);
}
// When parsing inline assembly we set the base register to a non-zero value // When parsing inline assembly we set the base register to a non-zero value
// as we don't know the actual value at this time. This is necessary to // as we don't know the actual value at this time. This is necessary to
// get the matching correct in some cases. // get the matching correct in some cases.
return X86Operand::CreateMem(/*SegReg*/0, Disp, /*BaseReg*/1, /*IndexReg*/0, return X86Operand::CreateMem(/*SegReg*/0, Disp, /*BaseReg*/1, /*IndexReg*/0,
/*Scale*/1, Start, End, Size, NeedSizeDir); /*Scale*/1, Start, End, Size, NeedSizeDir);
} }
}
/// Parse the '.' operator. /// Parse the '.' operator.
bool X86AsmParser::ParseIntelDotOperator(const MCExpr *Disp, bool X86AsmParser::ParseIntelDotOperator(const MCExpr *Disp,
@ -929,7 +951,8 @@ X86Operand *X86AsmParser::ParseIntelOffsetOfOperator(SMLoc Start) {
// register operand to ensure proper matching. Just pick a GPR based on // register operand to ensure proper matching. Just pick a GPR based on
// the size of a pointer. // the size of a pointer.
unsigned RegNo = is64BitMode() ? X86::RBX : X86::EBX; unsigned RegNo = is64BitMode() ? X86::RBX : X86::EBX;
return X86Operand::CreateReg(RegNo, Start, End, OffsetOfLoc); return X86Operand::CreateReg(RegNo, Start, End, /*GetAddress=*/true,
OffsetOfLoc);
} }
/// Parse the 'TYPE' operator. The TYPE operator returns the size of a C or /// Parse the 'TYPE' operator. The TYPE operator returns the size of a C or
@ -952,7 +975,9 @@ X86Operand *X86AsmParser::ParseIntelTypeOperator(SMLoc Start) {
// FIXME: The SemaLookup will fail if the name is anything other then an // FIXME: The SemaLookup will fail if the name is anything other then an
// identifier. // identifier.
// FIXME: Pass a valid SMLoc. // FIXME: Pass a valid SMLoc.
if (!SemaCallback->LookupInlineAsmIdentifier(Sym.getName(), NULL, Size)) bool IsVarDecl;
if (!SemaCallback->LookupInlineAsmIdentifier(Sym.getName(), NULL, Size,
IsVarDecl))
return ErrorOperand(Start, "Unable to lookup TYPE of expr!"); return ErrorOperand(Start, "Unable to lookup TYPE of expr!");
Size /= 8; // Size is in terms of bits, but we want bytes in the context. Size /= 8; // Size is in terms of bits, but we want bytes in the context.

View File

@ -61,3 +61,21 @@ entry:
; CHECK: .att_syntax ; CHECK: .att_syntax
; CHECK: {{## InlineAsm End|#NO_APP}} ; CHECK: {{## InlineAsm End|#NO_APP}}
} }
define void @t19_helper() nounwind {
entry:
ret void
}
define void @t19() nounwind {
entry:
call void asm sideeffect inteldialect "call $0", "r,~{dirflag},~{fpsr},~{flags}"(void ()* @t19_helper) nounwind
ret void
; CHECK: t19
; CHECK: movl $_t19_helper, %eax
; CHECK: {{## InlineAsm Start|#APP}}
; CHECK: .intel_syntax
; CHECK: call eax
; CHECK: .att_syntax
; CHECK: {{## InlineAsm End|#NO_APP}}
}