|
1 | 1 | #include "IR.h"
|
2 | 2 | #include "../Utils.h"
|
| 3 | +#include "types/PointerType.h" |
3 | 4 |
|
4 | 5 | IR::IR(std::string libName, std::string linkName, std::string objectName,
|
5 | 6 | std::string packageName, const LocationManager &locationManager)
|
@@ -111,26 +112,49 @@ llvm::raw_ostream &operator<<(llvm::raw_ostream &s, const IR &ir) {
|
111 | 112 | for (const auto &typeDef : ir.typeDefs) {
|
112 | 113 | if (ir.shouldOutput(typeDef)) {
|
113 | 114 | s << *typeDef;
|
| 115 | + if (typeDef->getType()) { |
| 116 | + auto *structOrUnion = |
| 117 | + dynamic_cast<StructOrUnion *>(typeDef->getType().get()); |
| 118 | + if (structOrUnion && |
| 119 | + structOrUnion->hasIllegalUsageOfOpaqueType()) { |
| 120 | + llvm::errs() |
| 121 | + << "Error: record " << structOrUnion->getName() |
| 122 | + << " has field of incomplete type. Declarations " |
| 123 | + "that use this type may not work properly.\n"; |
| 124 | + llvm::errs().flush(); |
| 125 | + } |
| 126 | + } |
114 | 127 | }
|
115 | 128 | }
|
116 | 129 |
|
117 | 130 | for (const auto &variable : ir.variables) {
|
118 |
| - s << *variable; |
| 131 | + if (!variable->hasIllegalUsageOfOpaqueType()) { |
| 132 | + s << *variable; |
| 133 | + } else { |
| 134 | + llvm::errs() << "Error: Variable " << variable->getName() |
| 135 | + << " is skipped because it has incomplete type.\n"; |
| 136 | + } |
119 | 137 | }
|
120 | 138 |
|
121 | 139 | for (const auto &varDefine : ir.varDefines) {
|
122 |
| - s << *varDefine; |
| 140 | + if (!varDefine->hasIllegalUsageOfOpaqueType()) { |
| 141 | + s << *varDefine; |
| 142 | + } else { |
| 143 | + llvm::errs() << "Error: Variable alias " << varDefine->getName() |
| 144 | + << " is skipped because it has incomplete type\n"; |
| 145 | + llvm::errs().flush(); |
| 146 | + } |
123 | 147 | }
|
124 | 148 |
|
125 | 149 | for (const auto &func : ir.functions) {
|
126 |
| - if (func->isLegalScalaNativeFunction()) { |
127 |
| - s << *func; |
128 |
| - } else { |
| 150 | + if (!func->isLegalScalaNativeFunction()) { |
129 | 151 | llvm::errs()
|
130 | 152 | << "Warning: Function " << func->getName()
|
131 | 153 | << " is skipped because Scala Native does not support "
|
132 | 154 | "passing structs and arrays by value.\n";
|
133 | 155 | llvm::errs().flush();
|
| 156 | + } else { |
| 157 | + s << *func; |
134 | 158 | }
|
135 | 159 | }
|
136 | 160 |
|
@@ -167,13 +191,14 @@ llvm::raw_ostream &operator<<(llvm::raw_ostream &s, const IR &ir) {
|
167 | 191 | s << "object " << ir.libName << "Helpers {\n";
|
168 | 192 |
|
169 | 193 | for (const auto &st : ir.structs) {
|
170 |
| - if (ir.shouldOutput(st) && st->hasHelperMethods()) { |
| 194 | + if (ir.shouldOutput(st) && st->hasHelperMethods() && |
| 195 | + !st->hasIllegalUsageOfOpaqueType()) { |
171 | 196 | s << "\n" << st->generateHelperClass();
|
172 | 197 | }
|
173 | 198 | }
|
174 | 199 |
|
175 | 200 | for (const auto &u : ir.unions) {
|
176 |
| - if (ir.shouldOutput(u)) { |
| 201 | + if (ir.shouldOutput(u) && !u->hasIllegalUsageOfOpaqueType()) { |
177 | 202 | s << "\n" << u->generateHelperClass();
|
178 | 203 | }
|
179 | 204 | }
|
@@ -446,5 +471,14 @@ bool IR::hasOutputtedDeclaration(
|
446 | 471 |
|
447 | 472 | template <typename T>
|
448 | 473 | bool IR::shouldOutput(const std::shared_ptr<T> &type) const {
|
449 |
| - return inMainFile(*type) || isTypeUsed(type, true); |
| 474 | + if (isTypeUsed(type, true)) { |
| 475 | + return true; |
| 476 | + } |
| 477 | + auto *typeDef = dynamic_cast<TypeDef *>(type.get()); |
| 478 | + if (typeDef) { |
| 479 | + /* unused typedefs from main file are not printed if they are aliases |
| 480 | + * for opaque types */ |
| 481 | + return !isAliasForOpaqueType(typeDef) && inMainFile(*typeDef); |
| 482 | + } |
| 483 | + return inMainFile(*type); |
450 | 484 | }
|
0 commit comments