blob: 454085bf2a52f3faf9dd9ca9e1b4fa094baeb37f [file] [log] [blame]
Chris Lattner4b009652007-07-25 00:24:17 +00001//===--- Builtins.cpp - Builtin function implementation -------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file was developed by Chris Lattner and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements various things for builtin functions.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/AST/Builtins.h"
15#include "clang/AST/ASTContext.h"
16#include "clang/Lex/IdentifierTable.h"
17#include "clang/Basic/TargetInfo.h"
18using namespace clang;
19
20static const Builtin::Info BuiltinInfo[] = {
21 { "not a builtin function", 0, 0 },
22#define BUILTIN(ID, TYPE, ATTRS) { #ID, TYPE, ATTRS },
23#include "clang/AST/Builtins.def"
24};
25
26const Builtin::Info &Builtin::Context::GetRecord(unsigned ID) const {
27 if (ID < Builtin::FirstTSBuiltin)
28 return BuiltinInfo[ID];
29 assert(ID - Builtin::FirstTSBuiltin < NumTSRecords && "Invalid builtin ID!");
30 return TSRecords[ID - Builtin::FirstTSBuiltin];
31}
32
33
34/// InitializeBuiltins - Mark the identifiers for all the builtins with their
35/// appropriate builtin ID # and mark any non-portable builtin identifiers as
36/// such.
37void Builtin::Context::InitializeBuiltins(IdentifierTable &Table,
38 const TargetInfo &Target) {
39 // Step #1: mark all target-independent builtins with their ID's.
40 for (unsigned i = Builtin::NotBuiltin+1; i != Builtin::FirstTSBuiltin; ++i)
41 Table.get(BuiltinInfo[i].Name).setBuiltinID(i);
42
43 // Step #2: handle target builtins.
44 std::vector<const char *> NonPortableBuiltins;
45 Target.getTargetBuiltins(TSRecords, NumTSRecords, NonPortableBuiltins);
46
47 // Step #2a: Register target-specific builtins.
48 for (unsigned i = 0, e = NumTSRecords; i != e; ++i)
49 Table.get(TSRecords[i].Name).setBuiltinID(i+Builtin::FirstTSBuiltin);
50
51 // Step #2b: Mark non-portable builtins as such.
52 for (unsigned i = 0, e = NonPortableBuiltins.size(); i != e; ++i)
53 Table.get(NonPortableBuiltins[i]).setNonPortableBuiltin(true);
54}
55
56/// DecodeTypeFromStr - This decodes one type descriptor from Str, advancing the
57/// pointer over the consumed characters. This returns the resultant type.
58static QualType DecodeTypeFromStr(const char *&Str, ASTContext &Context) {
59 // Modifiers.
60 bool Long = false, LongLong = false, Signed = false, Unsigned = false;
61
62 // Read the modifiers first.
63 bool Done = false;
64 while (!Done) {
65 switch (*Str++) {
66 default: Done = true; --Str; break;
67 case 'S':
68 assert(!Unsigned && "Can't use both 'S' and 'U' modifiers!");
69 assert(!Signed && "Can't use 'S' modifier multiple times!");
70 Signed = true;
71 break;
72 case 'U':
73 assert(!Signed && "Can't use both 'S' and 'U' modifiers!");
74 assert(!Unsigned && "Can't use 'S' modifier multiple times!");
75 Unsigned = true;
76 break;
77 case 'L':
78 assert(!LongLong && "Can't have LLL modifier");
79 if (Long)
80 LongLong = true;
81 else
82 Long = true;
83 break;
84 }
85 }
86
87 // Read the base type.
88 switch (*Str++) {
89 default: assert(0 && "Unknown builtin type letter!");
90 case 'v':
91 assert(!Long && !Signed && !Unsigned && "Bad modifiers used with 'f'!");
92 return Context.VoidTy;
93 case 'f':
94 assert(!Long && !Signed && !Unsigned && "Bad modifiers used with 'f'!");
95 return Context.FloatTy;
96 case 'd':
97 assert(!LongLong && !Signed && !Unsigned && "Bad modifiers used with 'd'!");
98 if (Long)
99 return Context.LongDoubleTy;
100 return Context.DoubleTy;
101 case 's':
102 assert(!LongLong && "Bad modifiers used with 's'!");
103 if (Unsigned)
104 return Context.UnsignedShortTy;
105 return Context.ShortTy;
106 //case 'i':
107 }
108}
109
110/// GetBuiltinType - Return the type for the specified builtin.
111QualType Builtin::Context::GetBuiltinType(unsigned id, ASTContext &Context)const{
112 const char *TypeStr = GetRecord(id).Type;
113
114 llvm::SmallVector<QualType, 8> ArgTypes;
115
116 QualType ResType = DecodeTypeFromStr(TypeStr, Context);
117 while (TypeStr[0] && TypeStr[0] != '.')
118 ArgTypes.push_back(DecodeTypeFromStr(TypeStr, Context));
119
120 assert((TypeStr[0] != '.' || TypeStr[1] == 0) &&
121 "'.' should only occur at end of builtin type list!");
122
123 return Context.getFunctionType(ResType, &ArgTypes[0], ArgTypes.size(),
124 TypeStr[0] == '.');
125}