blob: 597a0276966d00eb54628182bb48a99916942b8b [file] [log] [blame]
Anton Korobeynikov55bcea12010-01-10 12:58:08 +00001//===-- TargetAttributesSema.cpp - Encapsulate target attributes-*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file contains semantic analysis implementation for target-specific
11// attributes.
12//
13//===----------------------------------------------------------------------===//
14
15#include "Sema.h"
16#include "TargetAttributesSema.h"
17#include "clang/Basic/TargetInfo.h"
18#include "llvm/ADT/Triple.h"
19
20using namespace clang;
21
22TargetAttributesSema::~TargetAttributesSema() {}
23bool TargetAttributesSema::ProcessDeclAttribute(Scope *scope, Decl *D,
24 const AttributeList &Attr, Sema &S) const {
25 return false;
26}
27
28static void HandleMSP430InterruptAttr(Decl *d,
29 const AttributeList &Attr, Sema &S) {
30 // Check the attribute arguments.
31 if (Attr.getNumArgs() != 1) {
32 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
33 return;
34 }
35
36 // FIXME: Check for decl - it should be void ()(void).
37
38 Expr *NumParamsExpr = static_cast<Expr *>(Attr.getArg(0));
39 llvm::APSInt NumParams(32);
40 if (!NumParamsExpr->isIntegerConstantExpr(NumParams, S.Context)) {
41 S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
42 << "interrupt" << NumParamsExpr->getSourceRange();
43 return;
44 }
45
46 unsigned Num = NumParams.getLimitedValue(255);
47 if ((Num & 1) || Num > 30) {
48 S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
49 << "interrupt" << (int)NumParams.getSExtValue()
50 << NumParamsExpr->getSourceRange();
51 return;
52 }
53
54 d->addAttr(::new (S.Context) MSP430InterruptAttr(Num));
55 d->addAttr(::new (S.Context) UsedAttr());
56 }
57
58namespace {
59 class MSP430AttributesSema : public TargetAttributesSema {
60 public:
61 MSP430AttributesSema() { }
62 bool ProcessDeclAttribute(Scope *scope, Decl *D,
63 const AttributeList &Attr, Sema &S) const {
64 if (Attr.getName()->getName() == "interrupt") {
65 HandleMSP430InterruptAttr(D, Attr, S);
66 return true;
67 }
68 return false;
69 }
70 };
71}
72
Charles Davisbbc0aa52010-02-10 23:06:52 +000073static void HandleX86ForceAlignArgPointerAttr(Decl *D,
74 const AttributeList& Attr,
75 Sema &S) {
76 // Check the attribute arguments.
77 if (Attr.getNumArgs() != 0) {
78 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
79 return;
80 }
81
Charles Davisa90f7ca2010-02-11 23:57:08 +000082 // If we try to apply it to a function pointer, warn. This is a special
83 // instance of the warn_attribute_ignored warning that can be turned
84 // off with -Wno-force-align-arg-pointer.
Charles Davisbbc0aa52010-02-10 23:06:52 +000085 ValueDecl* VD = dyn_cast<ValueDecl>(D);
Charles Davisa90f7ca2010-02-11 23:57:08 +000086 if (VD && VD->getType()->isFunctionPointerType()) {
87 S.Diag(Attr.getLoc(), diag::warn_faap_attribute_ignored);
Charles Davisbbc0aa52010-02-10 23:06:52 +000088 return;
Charles Davisa90f7ca2010-02-11 23:57:08 +000089 }
Charles Davisbbc0aa52010-02-10 23:06:52 +000090 // Attribute can only be applied to function types.
Charles Davis42b38422010-02-10 23:26:12 +000091 if (!isa<FunctionDecl>(D)) {
Charles Davisbbc0aa52010-02-10 23:06:52 +000092 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
93 << Attr.getName() << /* function */0;
94 return;
95 }
96
97 D->addAttr(::new (S.Context) X86ForceAlignArgPointerAttr());
98}
99
Charles Davis163855f2010-02-16 18:27:26 +0000100static void HandleDLLImportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
101 // check the attribute arguments.
102 if (Attr.getNumArgs() != 0) {
103 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
104 return;
105 }
106
107 // Attribute can be applied only to functions or variables.
108 if (isa<VarDecl>(D)) {
109 D->addAttr(::new (S.Context) DLLImportAttr());
110 return;
111 }
112
113 FunctionDecl *FD = dyn_cast<FunctionDecl>(D);
114 if (!FD) {
115 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
116 << Attr.getName() << 2 /*variable and function*/;
117 return;
118 }
119
120 // Currently, the dllimport attribute is ignored for inlined functions.
121 // Warning is emitted.
122 if (FD->isInlineSpecified()) {
123 S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
124 return;
125 }
126
127 // The attribute is also overridden by a subsequent declaration as dllexport.
128 // Warning is emitted.
129 for (AttributeList *nextAttr = Attr.getNext(); nextAttr;
130 nextAttr = nextAttr->getNext()) {
131 if (nextAttr->getKind() == AttributeList::AT_dllexport) {
132 S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
133 return;
134 }
135 }
136
137 if (D->getAttr<DLLExportAttr>()) {
138 S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
139 return;
140 }
141
142 D->addAttr(::new (S.Context) DLLImportAttr());
143}
144
145static void HandleDLLExportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
146 // check the attribute arguments.
147 if (Attr.getNumArgs() != 0) {
148 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
149 return;
150 }
151
152 // Attribute can be applied only to functions or variables.
153 if (isa<VarDecl>(D)) {
154 D->addAttr(::new (S.Context) DLLExportAttr());
155 return;
156 }
157
158 FunctionDecl *FD = dyn_cast<FunctionDecl>(D);
159 if (!FD) {
160 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
161 << Attr.getName() << 2 /*variable and function*/;
162 return;
163 }
164
165 // Currently, the dllexport attribute is ignored for inlined functions, unless
166 // the -fkeep-inline-functions flag has been used. Warning is emitted;
167 if (FD->isInlineSpecified()) {
168 // FIXME: ... unless the -fkeep-inline-functions flag has been used.
169 S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllexport";
170 return;
171 }
172
173 D->addAttr(::new (S.Context) DLLExportAttr());
174}
175
Charles Davisbbc0aa52010-02-10 23:06:52 +0000176namespace {
177 class X86AttributesSema : public TargetAttributesSema {
178 public:
179 X86AttributesSema() { }
180 bool ProcessDeclAttribute(Scope *scope, Decl *D,
181 const AttributeList &Attr, Sema &S) const {
Charles Davis163855f2010-02-16 18:27:26 +0000182 const llvm::Triple &Triple(S.Context.Target.getTriple());
183 if (Triple.getOS() == llvm::Triple::Win32 ||
184 Triple.getOS() == llvm::Triple::MinGW32 ||
185 Triple.getOS() == llvm::Triple::MinGW64) {
186 switch (Attr.getKind()) {
187 case AttributeList::AT_dllimport: HandleDLLImportAttr(D, Attr, S);
188 return true;
189 case AttributeList::AT_dllexport: HandleDLLExportAttr(D, Attr, S);
190 return true;
191 default: break;
192 }
193 }
Charles Davisbbc0aa52010-02-10 23:06:52 +0000194 if (Attr.getName()->getName() == "force_align_arg_pointer") {
195 HandleX86ForceAlignArgPointerAttr(D, Attr, S);
196 return true;
197 }
198 return false;
199 }
200 };
201}
202
Anton Korobeynikov55bcea12010-01-10 12:58:08 +0000203const TargetAttributesSema &Sema::getTargetAttributesSema() const {
204 if (TheTargetAttributesSema)
205 return *TheTargetAttributesSema;
206
207 const llvm::Triple &Triple(Context.Target.getTriple());
208 switch (Triple.getArch()) {
209 default:
210 return *(TheTargetAttributesSema = new TargetAttributesSema);
211
212 case llvm::Triple::msp430:
213 return *(TheTargetAttributesSema = new MSP430AttributesSema);
Charles Davisbbc0aa52010-02-10 23:06:52 +0000214 case llvm::Triple::x86:
215 return *(TheTargetAttributesSema = new X86AttributesSema);
Anton Korobeynikov55bcea12010-01-10 12:58:08 +0000216 }
217}
218