slang_rs_export_type.cpp revision 116d3d2fcbde4970b07263deeeabc341ff8d76e9
1462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include <vector>
2462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
3462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_context.hpp"
4462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_type.hpp"
5462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_element.hpp"
6462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
7462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "llvm/Type.h"
8462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "llvm/DerivedTypes.h"
9462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
10462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "llvm/ADT/StringExtras.h"  /* for function llvm::utostr_32() */
11462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "llvm/Target/TargetData.h" /* for class llvm::TargetData and class llvm::StructLayout */
12462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
130a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao#include "clang/AST/RecordLayout.h"
140a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao
15462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaonamespace slang {
16462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
17462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportType ******************************/
18462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaobool RSExportType::NormalizeType(const Type*& T, llvm::StringRef& TypeName) {
19462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    llvm::SmallPtrSet<const Type*, 8> SPS = llvm::SmallPtrSet<const Type*, 8>();
20462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
211f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    if((T = RSExportType::TypeExportable(T, SPS)) == NULL)
22462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* TODO: warning the user: type not exportable */
23462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return false;
24462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
25462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* Get type name */
26462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    TypeName = RSExportType::GetTypeName(T);
27462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(TypeName.empty())
28462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* TODO: warning the user: the type is unnmaed */
29462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return false;
30462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
31462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return true;
32462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
33462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
34462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst Type* RSExportType::GetTypeOfDecl(const DeclaratorDecl* DD) {
35462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(DD && DD->getTypeSourceInfo()) {
36462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        QualType T = DD->getTypeSourceInfo()->getType();
37462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        if(T.isNull())
38462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return NULL;
39462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        else
40462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return T.getTypePtr();
41462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
42462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
43462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
44462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaollvm::StringRef RSExportType::GetTypeName(const Type* T) {
45462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    T = GET_CANONICAL_TYPE(T);
46462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(T == NULL)
47462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return llvm::StringRef();
48462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
49462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(T->getTypeClass()) {
50462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Builtin:
51462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
52462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const BuiltinType* BT = UNSAFE_CAST_TYPE(BuiltinType, T);
53462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
54462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            switch(BT->getKind()) {
55462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define SLANG_RS_SUPPORT_BUILTIN_TYPE(builtin_type, type) \
56462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                case builtin_type:  \
57462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    /* Compiler is smart enough to optimize following *big if branches* since they all become "constant comparison" after macro expansion */  \
58462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    if(type == RSExportPrimitiveType::DataTypeFloat32) return "float";  \
59462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeUnsigned8) return "uchar";   \
60462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeUnsigned16) return "ushort"; \
61462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeUnsigned32) return "uint";   \
62462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeSigned8) return "char";  \
63462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeSigned16) return "short";    \
64462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeSigned32) return "int";  \
651f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao                    else if(type == RSExportPrimitiveType::DataTypeBool) return "bool";  \
66462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    else assert(false && "Unknow data type of supported builtin");  \
67462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                break;
68462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_type_support.inc"
69462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
70462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                default: assert(false && "Unknow data type of the builtin"); break;
71462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
72462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
73462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
74462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
75462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Record:
76462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
77462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const RecordDecl* RD = T->getAsStructureType()->getDecl();
78462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            llvm::StringRef Name = RD->getName();
79462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(Name.empty()) {
801f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao                if(RD->getTypedefForAnonDecl() != NULL)
81462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    Name = RD->getTypedefForAnonDecl()->getName();
82462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
83462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                if(Name.empty())
84462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    /* Try to find a name from redeclaration (i.e. typedef) */
85462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    for(TagDecl::redecl_iterator RI = RD->redecls_begin();
86462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                        RI != RD->redecls_end();
87462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                        RI++)
88462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    {
89462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                        assert(*RI != NULL && "cannot be NULL object");
90462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
91462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                        Name = (*RI)->getName();
92462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                        if(!Name.empty())
93462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                            break;
94462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    }
95462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
96462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return Name;
97462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
98462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
99462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
100462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Pointer:
101462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
102462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* "*" plus pointee name */
103462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const Type* PT = GET_POINTEE_TYPE(T);
104462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            llvm::StringRef PointeeName;
105462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(NormalizeType(PT, PointeeName)) {
106462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                char* Name = new char[ 1 /* * */ + PointeeName.size() ];
107462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                Name[0] = '*';
108462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                memcpy(Name + 1, PointeeName.data(), PointeeName.size());
109462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return Name;
110462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
111462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
112462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
113462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
114462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::ExtVector:
115462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
116462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const ExtVectorType* EVT = UNSAFE_CAST_TYPE(ExtVectorType, T);
117462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return RSExportVectorType::GetTypeName(EVT);
118462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
119462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
120462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
121462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default:
122462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
123462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
124462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
125462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return llvm::StringRef();
126462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
127462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
128462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst Type* RSExportType::TypeExportable(const Type* T, llvm::SmallPtrSet<const Type*, 8>& SPS) {
129462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* Normailize first */
130462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((T = GET_CANONICAL_TYPE(T)) == NULL)
131462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
132462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
133462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(SPS.count(T))
134462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return T;
1351f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
136462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(T->getTypeClass()) {
137462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Builtin:
138462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
139462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const BuiltinType* BT = UNSAFE_CAST_TYPE(BuiltinType, T);
140462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
141462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            switch(BT->getKind()) {
142462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define SLANG_RS_SUPPORT_BUILTIN_TYPE(builtin_type, type) \
143462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                case builtin_type:
144462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_type_support.inc"
145462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    return T;
146462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                break;
147462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
148462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                default:
149462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    return NULL;
150462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                break;
151462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
152462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
153462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
154462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
155462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Record:
156462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
157462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(RSExportPrimitiveType::GetRSObjectType(T) != RSExportPrimitiveType::DataTypeUnknown)
158462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return T; /* RS object type, no further checks are needed */
159462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
160462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Check internal struct */
161462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const RecordDecl* RD = T->getAsStructureType()->getDecl();
162462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(RD != NULL)
163462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                RD = RD->getDefinition();
164462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
165462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Fast check */
1661f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            if(RD->hasFlexibleArrayMember() || RD->hasObjectMember())
167462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return NULL;
1681f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
169462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Insert myself into checking set */
170462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            SPS.insert(T);
171462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
172462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Check all element */
173462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            for(RecordDecl::field_iterator F = RD->field_begin();
174462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                F != RD->field_end();
175462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                F++)
176462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            {
177462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                const Type* FT = GetTypeOfDecl(*F);
178462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                FT = GET_CANONICAL_TYPE(FT);
179462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
1801f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao                if(!TypeExportable(FT, SPS))
181462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    /*  TODO: warning: unsupported field type */
182462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    return NULL;
183462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
184462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
185462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return T;
186462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
187462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
188462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
189462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Pointer:
190462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
191462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const PointerType* PT = UNSAFE_CAST_TYPE(PointerType, T);
192462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const Type* PointeeType = GET_POINTEE_TYPE(PT);
193462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
194462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if((PointeeType->getTypeClass() != Type::Pointer) && (TypeExportable(PointeeType, SPS) == NULL) )
195462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return NULL;
196462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return T;
197462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
198462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
199462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
200462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::ExtVector:
201462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
202462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const ExtVectorType* EVT = UNSAFE_CAST_TYPE(ExtVectorType, T);
203462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Check element numbers */
204462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(EVT->getNumElements() < 2 || EVT->getNumElements() > 4)  /* only support vector with size 2, 3 and 4 */
205462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return NULL;
206462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
207462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Check base element type */
208462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const Type* ElementType = GET_EXT_VECTOR_ELEMENT_TYPE(EVT);
209462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
210462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if((ElementType->getTypeClass() != Type::Builtin) || (TypeExportable(ElementType, SPS) == NULL))
211462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                return NULL;
212462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return T;
213462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
214462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
215462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
216462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default:
217462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return NULL;
218462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
219462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
220462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
221462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
222462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportType* RSExportType::Create(RSContext* Context, const Type* T, const llvm::StringRef& TypeName) {
2231f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    /*
2241f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao     * Lookup the context to see whether the type was processed before.
2251f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao     *  Newly create RSExportType will insert into context in RSExportType::RSExportType()
226462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao     */
227462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RSContext::export_type_iterator ETI = Context->findExportType(TypeName);
228462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
2291f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    if(ETI != Context->export_types_end())
230462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return ETI->second;
231462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
232462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RSExportType* ET = NULL;
233462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(T->getTypeClass()) {
234462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Record:
235462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(RSExportPrimitiveType::GetRSObjectType(TypeName) != RSExportPrimitiveType::DataTypeUnknown)
236462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                ET = RSExportPrimitiveType::Create(Context, T, TypeName);
237462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else
238462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                ET = RSExportRecordType::Create(Context, T->getAsStructureType(), TypeName);
239462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
240462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
241462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Builtin:
242462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            ET = RSExportPrimitiveType::Create(Context, T, TypeName);
243462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
244462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
245462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Pointer:
246462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            ET = RSExportPointerType::Create(Context, UNSAFE_CAST_TYPE(PointerType, T), TypeName);
247462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* free the name (allocated in RSExportType::GetTypeName) */
248462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            delete [] TypeName.data();
249462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
250462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
251462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::ExtVector:
252462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            ET = RSExportVectorType::Create(Context, UNSAFE_CAST_TYPE(ExtVectorType, T), TypeName);
253462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
254462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
255462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default:
256462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* TODO: warning: type is not exportable */
257462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            printf("RSExportType::Create : type '%s' is not exportable\n", T->getTypeClassName());
258462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
259462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
260462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
261462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return ET;
262462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
2631f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
264462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportType* RSExportType::Create(RSContext* Context, const Type* T) {
265462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    llvm::StringRef TypeName;
266462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(NormalizeType(T, TypeName))
267462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return Create(Context, T, TypeName);
268462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
269462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
270462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
271462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
272462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportType* RSExportType::CreateFromDecl(RSContext* Context, const VarDecl* VD) {
273462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return RSExportType::Create(Context, GetTypeOfDecl(VD));
274462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
275462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
276462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaosize_t RSExportType::GetTypeStoreSize(const RSExportType* ET) {
277462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return ET->getRSContext()->getTargetData()->getTypeStoreSize(ET->getLLVMType());
278462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
279462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
280462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaosize_t RSExportType::GetTypeAllocSize(const RSExportType* ET) {
2810a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao    if(ET->getClass() == RSExportType::ExportClassRecord)
2820a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao        return static_cast<const RSExportRecordType*>(ET)->getAllocSize();
2830a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao    else
2840a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao        return ET->getRSContext()->getTargetData()->getTypeAllocSize(ET->getLLVMType());
285462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
286462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
2871f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei LiaoRSExportType::RSExportType(RSContext* Context, const llvm::StringRef& Name) :
2881f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    mContext(Context),
2891f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    mName(Name.data(), Name.size()), /* make a copy on Name since data of @Name which is stored in ASTContext will be destroyed later */
290462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    mLLVMType(NULL)
2911f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao{
292462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* TODO: need to check whether the insertion is successful or not */
293462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    Context->insertExportType(Name, this);
2941f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    return;
295462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
296462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
297462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportPrimitiveType ******************************/
298462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType::RSObjectTypeMapTy* RSExportPrimitiveType::RSObjectTypeMap = NULL;
299462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaollvm::Type* RSExportPrimitiveType::RSObjectLLVMType = NULL;
300462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
301462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaobool RSExportPrimitiveType::IsPrimitiveType(const Type* T) {
302462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((T != NULL) && (T->getTypeClass() == Type::Builtin))
303462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return true;
304462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
305462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return false;
306462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
307462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
308462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType::DataType RSExportPrimitiveType::GetRSObjectType(const llvm::StringRef& TypeName) {
309462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(TypeName.empty())
310462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return DataTypeUnknown;
311462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
312462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(RSObjectTypeMap == NULL) {
313462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        RSObjectTypeMap = new RSObjectTypeMapTy(16);
314462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
315462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define USE_ELEMENT_DATA_TYPE
316462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define DEF_RS_OBJECT_TYPE(type, name)  \
317462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        RSObjectTypeMap->GetOrCreateValue(name, GET_ELEMENT_DATA_TYPE(type));
318462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_element_support.inc"
319462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
320462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
321462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RSObjectTypeMapTy::const_iterator I = RSObjectTypeMap->find(TypeName);
322462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(I == RSObjectTypeMap->end())
323462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return DataTypeUnknown;
324462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
325462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return I->getValue();
326462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
327462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
328462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType::DataType RSExportPrimitiveType::GetRSObjectType(const Type* T) {
329462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    T = GET_CANONICAL_TYPE(T);
330462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((T == NULL) || (T->getTypeClass() != Type::Record))
331462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return DataTypeUnknown;
332462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
333462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return GetRSObjectType( RSExportType::GetTypeName(T) );
334462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
335462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
3361f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liaoconst size_t RSExportPrimitiveType::SizeOfDataTypeInBits[RSExportPrimitiveType::DataTypeMax + 1] = {
337462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    0,
3381f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeFloat16 */
3391f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeFloat32 */
3401f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    64, /* DataTypeFloat64 */
3411f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    8, /* DataTypeSigned8 */
3421f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeSigned16 */
3431f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeSigned32 */
3441f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    64, /* DataTypeSigned64 */
3451f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    8, /* DataTypeUnsigned8 */
3461f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeUnsigned16 */
3471f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeUnsigned32 */
3481f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    64, /* DataTypeUnSigned64 */
3491f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
3501f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeUnsigned565 */
3511f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeUnsigned5551 */
3521f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    16, /* DataTypeUnsigned4444 */
3531f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
3541f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    1, /* DataTypeBool */
3551f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
3561f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSElement */
3571f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSType */
3581f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSAllocation */
3591f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSSampler */
3601f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSScript */
3611f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSMesh */
3621f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSProgramFragment */
3631f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSProgramVertex */
3641f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSProgramRaster */
3651f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSProgramStore */
3661f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    32, /* DataTypeRSFont */
367116d3d2fcbde4970b07263deeeabc341ff8d76e9Shih-wei Liao    128, /* DataTypeRSMatrix2x2 */
368116d3d2fcbde4970b07263deeeabc341ff8d76e9Shih-wei Liao    288, /* DataTypeRSMatrix3x3 */
369116d3d2fcbde4970b07263deeeabc341ff8d76e9Shih-wei Liao    512, /* DataTypeRSMatrix4x4 */
370462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    0
371462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao};
372462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
3731f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liaosize_t RSExportPrimitiveType::GetSizeInBits(const RSExportPrimitiveType* EPT) {
3741f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    assert(((EPT->getType() >= DataTypeFloat32) && (EPT->getType() < DataTypeMax)) && "RSExportPrimitiveType::GetSizeInBits : unknown data type");
3751f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    return SizeOfDataTypeInBits[ static_cast<int>(EPT->getType()) ];
376462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
377462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
378462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType::DataType RSExportPrimitiveType::GetDataType(const Type* T) {
379462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(T == NULL)
380462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return DataTypeUnknown;
381462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
382462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(T->getTypeClass()) {
383462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Builtin:
384462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        {
385462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            const BuiltinType* BT = UNSAFE_CAST_TYPE(BuiltinType, T);
386462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            switch(BT->getKind()) {
387462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define SLANG_RS_SUPPORT_BUILTIN_TYPE(builtin_type, type) \
388462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                case builtin_type:  \
389462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    return type;  \
390462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                break;
391462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_type_support.inc"
392462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
393462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                /* The size of types Long, ULong and WChar depend on platform so we abandon the support to them */
394462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                /* Type of its size exceeds 32 bits (e.g. int64_t, double, etc.) does not support */
395462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
396462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                default:
397462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    /* TODO: warning the user: unsupported type */
398462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                    printf("RSExportPrimitiveType::GetDataType : built-in type has no corresponding data type for built-in type");
399462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                break;
400462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            }
401462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
402462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
403462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
404462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case Type::Record:
405462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* must be RS object type */
406462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            return RSExportPrimitiveType::GetRSObjectType(T);
407462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
408462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
409462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default:
410462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            printf("RSExportPrimitiveType::GetDataType : type '%s' is not supported primitive type", T->getTypeClassName());
411462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
412462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
413462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
414462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return DataTypeUnknown;
415462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
416462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
417462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType* RSExportPrimitiveType::Create(RSContext* Context, const Type* T, const llvm::StringRef& TypeName, DataKind DK, bool Normalized) {
418462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    DataType DT = GetDataType(T);
419462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
420462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((DT == DataTypeUnknown) || TypeName.empty())
421462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
422462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
423462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return new RSExportPrimitiveType(Context, TypeName, DT, DK, Normalized);
424462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
425462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
426462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPrimitiveType* RSExportPrimitiveType::Create(RSContext* Context, const Type* T, DataKind DK) {
427462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    llvm::StringRef TypeName;
428462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(RSExportType::NormalizeType(T, TypeName) && IsPrimitiveType(T))
429462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return Create(Context, T, TypeName, DK);
430462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
431462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
432462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
433462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
4341f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei LiaoRSExportType::ExportClass RSExportPrimitiveType::getClass() const {
4351f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    return RSExportType::ExportClassPrimitive;
436462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
437462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
438462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst llvm::Type* RSExportPrimitiveType::convertToLLVMType() const {
439462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    llvm::LLVMContext& C = getRSContext()->getLLVMContext();
440462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
441462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(isRSObjectType()) {
442462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* struct { int* p; } __attribute__((packed, aligned(pointer_size))) which is <{ [1 x i32] }> in LLVM */
443462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        if(RSObjectLLVMType == NULL) {
444462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            std::vector<const llvm::Type*> Elements;
445462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            Elements.push_back( llvm::ArrayType::get(llvm::Type::getInt32Ty(C), 1) );
446462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            RSObjectLLVMType = llvm::StructType::get(C, Elements, true);
447462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        }
448462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return RSObjectLLVMType;
449462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
450462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
451462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(mType) {
4521f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao        case DataTypeFloat32:
4531f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            return llvm::Type::getFloatTy(C);
454462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
455462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
456462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeSigned8:
457462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned8:
4581f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            return llvm::Type::getInt8Ty(C);
459462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
460462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
461462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeSigned16:
462462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned16:
463462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned565:
464462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned5551:
465462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned4444:
4661f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            return llvm::Type::getInt16Ty(C);
467462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
468462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
469462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeSigned32:
470462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case DataTypeUnsigned32:
4711f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            return llvm::Type::getInt32Ty(C);
4721f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao        break;
4731f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
4741f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao        case DataTypeBool:
4751f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            return llvm::Type::getInt1Ty(C);
476462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
477462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
478462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default:
479462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            assert(false && "Unknown data type");
480462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
481462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
482462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
483462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return NULL;
484462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
485462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
486462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportPointerType ******************************/
487462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
488462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst Type* RSExportPointerType::IntegerType = NULL;
489462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
490462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportPointerType* RSExportPointerType::Create(RSContext* Context, const PointerType* PT, const llvm::StringRef& TypeName) {
491462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const Type* PointeeType = GET_POINTEE_TYPE(PT);
492462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const RSExportType* PointeeET;
493462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
494462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(PointeeType->getTypeClass() != Type::Pointer) {
495462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        PointeeET = RSExportType::Create(Context, PointeeType);
496462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    } else {
497462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* Double or higher dimension of pointer, export as int* */
498462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        assert(IntegerType != NULL && "Built-in integer type is not set");
499462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        PointeeET = RSExportPrimitiveType::Create(Context, IntegerType);
500462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
501462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
502462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(PointeeET == NULL) {
503462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        printf("Failed to create type for pointee");
504462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
505462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
506462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
507462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return new RSExportPointerType(Context, TypeName, PointeeET);
508462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
509462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
5101f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei LiaoRSExportType::ExportClass RSExportPointerType::getClass() const {
511462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return RSExportType::ExportClassPointer;
512462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
513462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
514462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst llvm::Type* RSExportPointerType::convertToLLVMType() const {
515462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const llvm::Type* PointeeType = mPointeeType->getLLVMType();
516462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return llvm::PointerType::getUnqual(PointeeType);
517462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
518462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
519462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportVectorType ******************************/
5201f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liaoconst char* RSExportVectorType::VectorTypeNameStore[][3] = {
521462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 0 */ { "char2",      "char3",    "char4" },
522462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 1 */ { "uchar2",     "uchar3",   "uchar4" },
523462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 2 */ { "short2",     "short3",   "short4" },
524462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 3 */ { "ushort2",    "ushort3",  "ushort4" },
525462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 4 */ { "int2",       "int3",     "int4" },
526462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 5 */ { "uint2",      "uint3",    "uint4" },
527462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* 6 */ { "float2",     "float3",   "float4" },
5281f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao};
529462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
530462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaollvm::StringRef RSExportVectorType::GetTypeName(const ExtVectorType* EVT) {
531462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const Type* ElementType = GET_EXT_VECTOR_ELEMENT_TYPE(EVT);
532462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
533462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((ElementType->getTypeClass() != Type::Builtin))
534462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return llvm::StringRef();
535462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
536462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const BuiltinType* BT = UNSAFE_CAST_TYPE(BuiltinType, ElementType);
537462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const char** BaseElement = NULL;
538462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
539462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    switch(BT->getKind()) {
540462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define SLANG_RS_SUPPORT_BUILTIN_TYPE(builtin_type, type) \
541462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        case builtin_type:  \
542462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            /* Compiler is smart enough to optimize following *big if branches* since they all become "constant comparison" after macro expansion */  \
543462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            if(type == RSExportPrimitiveType::DataTypeSigned8) BaseElement = VectorTypeNameStore[0];    \
544462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeUnsigned8) BaseElement = VectorTypeNameStore[1]; \
545462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeSigned16) BaseElement = VectorTypeNameStore[2];  \
546462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeUnsigned16) BaseElement = VectorTypeNameStore[3];    \
547462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeSigned32) BaseElement = VectorTypeNameStore[4];  \
548462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeUnsigned32) BaseElement = VectorTypeNameStore[5];    \
549462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeFloat32) BaseElement = VectorTypeNameStore[6];   \
5501f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao            else if(type == RSExportPrimitiveType::DataTypeBool) BaseElement = VectorTypeNameStore[0];   \
551462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        break;
552462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#include "slang_rs_export_type_support.inc"
553462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
554462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        default: return llvm::StringRef(); break;
555462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
556462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
557462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if((BaseElement != NULL) && (EVT->getNumElements() > 1) && (EVT->getNumElements() <= 4))
558462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return BaseElement[EVT->getNumElements() - 2];
559462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
560462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return llvm::StringRef();
561462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
562462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
563462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportVectorType* RSExportVectorType::Create(RSContext* Context, const ExtVectorType* EVT, const llvm::StringRef& TypeName, DataKind DK, bool Normalized) {
564462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    assert(EVT != NULL && EVT->getTypeClass() == Type::ExtVector);
565462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
566462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const Type* ElementType = GET_EXT_VECTOR_ELEMENT_TYPE(EVT);
567462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RSExportPrimitiveType::DataType DT = RSExportPrimitiveType::GetDataType(ElementType);
568462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
5691f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    if(DT != RSExportPrimitiveType::DataTypeUnknown)
570462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return new RSExportVectorType(Context, TypeName, DT, DK, Normalized, EVT->getNumElements());
571462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    else
572462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        printf("RSExportVectorType::Create : unsupported base element type\n");
573462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
574462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
5751f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei LiaoRSExportType::ExportClass RSExportVectorType::getClass() const {
5761f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao    return RSExportType::ExportClassVector;
577462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
578462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
579462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst llvm::Type* RSExportVectorType::convertToLLVMType() const {
580462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const llvm::Type* ElementType = RSExportPrimitiveType::convertToLLVMType();
581462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return llvm::VectorType::get(ElementType, getNumElement());
582462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
583462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
584462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportRecordType ******************************/
585462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportRecordType* RSExportRecordType::Create(RSContext* Context, const RecordType* RT, const llvm::StringRef& TypeName, bool mIsArtificial) {
586462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    assert(RT != NULL && RT->getTypeClass() == Type::Record);
587462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
588462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const RecordDecl* RD = RT->getDecl();
589462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    assert(RD->isStruct());
590462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
591462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RD = RD->getDefinition();
592462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    if(RD == NULL) {
593462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* TODO: warning: actual struct definition isn't declared in this moudle */
594462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        printf("RSExportRecordType::Create : this struct is not defined in this module.");
595462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        return NULL;
596462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
597462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
598462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    RSExportRecordType* ERT = new RSExportRecordType(Context, TypeName, RD->hasAttr<PackedAttr>(), mIsArtificial);
599462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    unsigned int Index = 0;
600462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
601462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    for(RecordDecl::field_iterator fit = RD->field_begin();
602462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        fit != RD->field_end();
603462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        fit++, Index++)
604462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    {
605462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#define FAILED_CREATE_FIELD(err)    do {    \
606462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                                        if(*err) \
607462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                                            printf("RSExportRecordType::Create : failed to create field (%s)\n", err);   \
608462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                                        delete ERT;  \
609462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                                        return NULL;    \
610462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao                                    } while(false)
611462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* FIXME: All fields should be primitive type */
612462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        assert((*fit)->getKind() == Decl::Field);
613462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        FieldDecl* FD = *fit;
614462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
615462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* We don't support bit field TODO: allow bitfield with size 8, 16, 32 */
616462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        if(FD->isBitField())
617462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            FAILED_CREATE_FIELD("bit field is not supported");
618462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
619462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        /* Type */
620462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        RSExportType* ET = RSExportElement::CreateFromDecl(Context, FD);
621462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
622462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        if(ET != NULL)
623462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            ERT->mFields.push_back( new Field(ET, FD->getName(), ERT, Index) );
624462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        else
625462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao            FAILED_CREATE_FIELD(FD->getName().str().c_str());
626462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao#undef FAILED_CREATE_FIELD
627462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
628462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
6290a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao
6300a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao    const ASTRecordLayout &ASTRL = Context->getASTContext()->getASTRecordLayout(RD);
6310a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao    ERT->AllocSize = (ASTRL.getSize() > ASTRL.getDataSize()) ? (ASTRL.getSize() >> 3) : (ASTRL.getDataSize() >> 3);
6320a3f20ec28ed6f5ae1ed5d61f6b6e3e577f7f5d1Shih-wei Liao
633462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return ERT;
634462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
635462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
636462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei LiaoRSExportType::ExportClass RSExportRecordType::getClass() const {
637462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return RSExportType::ExportClassRecord;
638462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
639462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
640462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaoconst llvm::Type* RSExportRecordType::convertToLLVMType() const {
641462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    std::vector<const llvm::Type*> FieldTypes;
642462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
643462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    for(const_field_iterator FI = fields_begin();
644462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        FI != fields_end();
645462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        FI++)
646462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    {
647462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        const Field* F = *FI;
648462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        const RSExportType* FET = F->getType();
6491f0d88fbff28e4e2dd563d93c8fe0047381c09ccShih-wei Liao
650462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao        FieldTypes.push_back(FET->getLLVMType());
651462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    }
652462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
653462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return llvm::StructType::get(getRSContext()->getLLVMContext(), FieldTypes, mIsPacked);
654462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
655462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
656462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao/****************************** RSExportRecordType::Field ******************************/
657462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liaosize_t RSExportRecordType::Field::getOffsetInParent() const {
658462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    /* Struct layout obtains below will be cached by LLVM */
659462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    const llvm::StructLayout* SL = mParent->getRSContext()->getTargetData()->getStructLayout(static_cast<const llvm::StructType*>(mParent->getLLVMType()));
660462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao    return SL->getElementOffset(mIndex);
661462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}
662462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao
663462aefd62cc646d2ff753c1d003ef3cd7bbea26Shih-wei Liao}   /* namespace slang */
664