| 1 | //===-- Options.cpp -------------------------------------------------------===// |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #include "Options.h" |
| 10 | #include "clang/Basic/DiagnosticIDs.h" |
| 11 | #include "clang/Driver/Driver.h" |
| 12 | #include "clang/InstallAPI/DirectoryScanner.h" |
| 13 | #include "clang/InstallAPI/FileList.h" |
| 14 | #include "clang/InstallAPI/HeaderFile.h" |
| 15 | #include "clang/InstallAPI/InstallAPIDiagnostic.h" |
| 16 | #include "llvm/BinaryFormat/Magic.h" |
| 17 | #include "llvm/Support/JSON.h" |
| 18 | #include "llvm/Support/Program.h" |
| 19 | #include "llvm/TargetParser/Host.h" |
| 20 | #include "llvm/TextAPI/DylibReader.h" |
| 21 | #include "llvm/TextAPI/TextAPIError.h" |
| 22 | #include "llvm/TextAPI/TextAPIReader.h" |
| 23 | #include "llvm/TextAPI/TextAPIWriter.h" |
| 24 | |
| 25 | using namespace llvm; |
| 26 | using namespace llvm::opt; |
| 27 | using namespace llvm::MachO; |
| 28 | |
| 29 | namespace drv = clang::driver::options; |
| 30 | |
| 31 | namespace clang { |
| 32 | namespace installapi { |
| 33 | |
| 34 | #define OPTTABLE_STR_TABLE_CODE |
| 35 | #include "InstallAPIOpts.inc" |
| 36 | #undef OPTTABLE_STR_TABLE_CODE |
| 37 | |
| 38 | #define OPTTABLE_PREFIXES_TABLE_CODE |
| 39 | #include "InstallAPIOpts.inc" |
| 40 | #undef OPTTABLE_PREFIXES_TABLE_CODE |
| 41 | |
| 42 | #define OPTTABLE_PREFIXES_UNION_CODE |
| 43 | #include "InstallAPIOpts.inc" |
| 44 | #undef OPTTABLE_PREFIXES_UNION_CODE |
| 45 | |
| 46 | /// Create table mapping all options defined in InstallAPIOpts.td. |
| 47 | static constexpr OptTable::Info InfoTable[] = { |
| 48 | #define OPTION(...) LLVM_CONSTRUCT_OPT_INFO(__VA_ARGS__), |
| 49 | #include "InstallAPIOpts.inc" |
| 50 | #undef OPTION |
| 51 | }; |
| 52 | |
| 53 | namespace { |
| 54 | |
| 55 | /// \brief Create OptTable class for parsing actual command line arguments. |
| 56 | class DriverOptTable : public opt::PrecomputedOptTable { |
| 57 | public: |
| 58 | DriverOptTable() |
| 59 | : PrecomputedOptTable(OptionStrTable, OptionPrefixesTable, InfoTable, |
| 60 | OptionPrefixesUnion) {} |
| 61 | }; |
| 62 | |
| 63 | } // end anonymous namespace. |
| 64 | |
| 65 | static llvm::opt::OptTable *createDriverOptTable() { |
| 66 | return new DriverOptTable(); |
| 67 | } |
| 68 | |
| 69 | /// Parse JSON input into argument list. |
| 70 | /// |
| 71 | /* Expected input format. |
| 72 | * { "label" : ["-ClangArg1", "-ClangArg2"] } |
| 73 | */ |
| 74 | /// |
| 75 | /// Input is interpreted as "-Xlabel ClangArg1 -XLabel ClangArg2". |
| 76 | static Expected<llvm::opt::InputArgList> |
| 77 | getArgListFromJSON(const StringRef Input, llvm::opt::OptTable *Table, |
| 78 | std::vector<std::string> &Storage) { |
| 79 | using namespace json; |
| 80 | Expected<Value> ValOrErr = json::parse(JSON: Input); |
| 81 | if (!ValOrErr) |
| 82 | return ValOrErr.takeError(); |
| 83 | |
| 84 | const Object *Root = ValOrErr->getAsObject(); |
| 85 | if (!Root) |
| 86 | return llvm::opt::InputArgList(); |
| 87 | |
| 88 | for (const auto &KV : *Root) { |
| 89 | const Array *ArgList = KV.getSecond().getAsArray(); |
| 90 | std::string Label = "-X" + KV.getFirst().str(); |
| 91 | if (!ArgList) |
| 92 | return make_error<TextAPIError>(Args: TextAPIErrorCode::InvalidInputFormat); |
| 93 | for (auto Arg : *ArgList) { |
| 94 | std::optional<StringRef> ArgStr = Arg.getAsString(); |
| 95 | if (!ArgStr) |
| 96 | return make_error<TextAPIError>(Args: TextAPIErrorCode::InvalidInputFormat); |
| 97 | Storage.emplace_back(args&: Label); |
| 98 | Storage.emplace_back(args&: *ArgStr); |
| 99 | } |
| 100 | } |
| 101 | |
| 102 | std::vector<const char *> CArgs(Storage.size()); |
| 103 | for (StringRef Str : Storage) |
| 104 | CArgs.emplace_back(args: Str.data()); |
| 105 | |
| 106 | unsigned MissingArgIndex, MissingArgCount; |
| 107 | return Table->ParseArgs(Args: CArgs, MissingArgIndex, MissingArgCount); |
| 108 | } |
| 109 | |
| 110 | bool Options::processDriverOptions(InputArgList &Args) { |
| 111 | // Handle inputs. |
| 112 | for (const StringRef Path : Args.getAllArgValues(Id: drv::OPT_INPUT)) { |
| 113 | // Assume any input that is not a directory is a filelist. |
| 114 | // InstallAPI does not accept multiple directories, so retain the last one. |
| 115 | if (FM->getOptionalDirectoryRef(DirName: Path)) |
| 116 | DriverOpts.InputDirectory = Path.str(); |
| 117 | else |
| 118 | DriverOpts.FileLists.emplace_back(args: Path.str()); |
| 119 | } |
| 120 | |
| 121 | // Handle output. |
| 122 | SmallString<PATH_MAX> OutputPath; |
| 123 | if (auto *Arg = Args.getLastArg(Ids: drv::OPT_o)) { |
| 124 | OutputPath = Arg->getValue(); |
| 125 | if (OutputPath != "-" ) |
| 126 | FM->makeAbsolutePath(Path&: OutputPath); |
| 127 | DriverOpts.OutputPath = std::string(OutputPath); |
| 128 | } |
| 129 | if (DriverOpts.OutputPath.empty()) { |
| 130 | Diags->Report(DiagID: diag::err_no_output_file); |
| 131 | return false; |
| 132 | } |
| 133 | |
| 134 | // Do basic error checking first for mixing -target and -arch options. |
| 135 | auto *ArgArch = Args.getLastArgNoClaim(Ids: drv::OPT_arch); |
| 136 | auto *ArgTarget = Args.getLastArgNoClaim(Ids: drv::OPT_target); |
| 137 | auto *ArgTargetVariant = |
| 138 | Args.getLastArgNoClaim(Ids: drv::OPT_darwin_target_variant); |
| 139 | if (ArgArch && (ArgTarget || ArgTargetVariant)) { |
| 140 | Diags->Report(DiagID: clang::diag::err_drv_argument_not_allowed_with) |
| 141 | << ArgArch->getAsString(Args) |
| 142 | << (ArgTarget ? ArgTarget : ArgTargetVariant)->getAsString(Args); |
| 143 | return false; |
| 144 | } |
| 145 | |
| 146 | auto *ArgMinTargetOS = Args.getLastArgNoClaim(Ids: drv::OPT_mtargetos_EQ); |
| 147 | if ((ArgTarget || ArgTargetVariant) && ArgMinTargetOS) { |
| 148 | Diags->Report(DiagID: clang::diag::err_drv_cannot_mix_options) |
| 149 | << ArgTarget->getAsString(Args) << ArgMinTargetOS->getAsString(Args); |
| 150 | return false; |
| 151 | } |
| 152 | |
| 153 | // Capture target triples first. |
| 154 | if (ArgTarget) { |
| 155 | for (const Arg *A : Args.filtered(Ids: drv::OPT_target)) { |
| 156 | A->claim(); |
| 157 | llvm::Triple TargetTriple(A->getValue()); |
| 158 | Target TAPITarget = Target(TargetTriple); |
| 159 | if ((TAPITarget.Arch == AK_unknown) || |
| 160 | (TAPITarget.Platform == PLATFORM_UNKNOWN)) { |
| 161 | Diags->Report(DiagID: clang::diag::err_drv_unsupported_opt_for_target) |
| 162 | << "installapi" << TargetTriple.str(); |
| 163 | return false; |
| 164 | } |
| 165 | DriverOpts.Targets[TAPITarget] = TargetTriple; |
| 166 | } |
| 167 | } |
| 168 | |
| 169 | // Capture target variants. |
| 170 | DriverOpts.Zippered = ArgTargetVariant != nullptr; |
| 171 | for (Arg *A : Args.filtered(Ids: drv::OPT_darwin_target_variant)) { |
| 172 | A->claim(); |
| 173 | Triple Variant(A->getValue()); |
| 174 | if (Variant.getVendor() != Triple::Apple) { |
| 175 | Diags->Report(DiagID: diag::err_unsupported_vendor) |
| 176 | << Variant.getVendorName() << A->getAsString(Args); |
| 177 | return false; |
| 178 | } |
| 179 | |
| 180 | switch (Variant.getOS()) { |
| 181 | default: |
| 182 | Diags->Report(DiagID: diag::err_unsupported_os) |
| 183 | << Variant.getOSName() << A->getAsString(Args); |
| 184 | return false; |
| 185 | case Triple::MacOSX: |
| 186 | case Triple::IOS: |
| 187 | break; |
| 188 | } |
| 189 | |
| 190 | switch (Variant.getEnvironment()) { |
| 191 | default: |
| 192 | Diags->Report(DiagID: diag::err_unsupported_environment) |
| 193 | << Variant.getEnvironmentName() << A->getAsString(Args); |
| 194 | return false; |
| 195 | case Triple::UnknownEnvironment: |
| 196 | case Triple::MacABI: |
| 197 | break; |
| 198 | } |
| 199 | |
| 200 | Target TAPIVariant(Variant); |
| 201 | // See if there is a matching --target option for this --target-variant |
| 202 | // option. |
| 203 | auto It = find_if(Range&: DriverOpts.Targets, P: [&](const auto &T) { |
| 204 | return (T.first.Arch == TAPIVariant.Arch) && |
| 205 | (T.first.Platform != PlatformType::PLATFORM_UNKNOWN); |
| 206 | }); |
| 207 | |
| 208 | if (It == DriverOpts.Targets.end()) { |
| 209 | Diags->Report(DiagID: diag::err_no_matching_target) << Variant.str(); |
| 210 | return false; |
| 211 | } |
| 212 | |
| 213 | DriverOpts.Targets[TAPIVariant] = Variant; |
| 214 | } |
| 215 | |
| 216 | DriverOpts.Verbose = Args.hasArgNoClaim(Ids: drv::OPT_v); |
| 217 | |
| 218 | return true; |
| 219 | } |
| 220 | |
| 221 | bool Options::processInstallAPIXOptions(InputArgList &Args) { |
| 222 | for (arg_iterator It = Args.begin(), End = Args.end(); It != End; ++It) { |
| 223 | Arg *A = *It; |
| 224 | if (A->getOption().matches(ID: OPT_Xarch__)) { |
| 225 | if (!processXarchOption(Args, Curr: It)) |
| 226 | return false; |
| 227 | continue; |
| 228 | } else if (A->getOption().matches(ID: OPT_Xplatform__)) { |
| 229 | if (!processXplatformOption(Args, Curr: It)) |
| 230 | return false; |
| 231 | continue; |
| 232 | } else if (A->getOption().matches(ID: OPT_Xproject)) { |
| 233 | if (!processXprojectOption(Args, Curr: It)) |
| 234 | return false; |
| 235 | continue; |
| 236 | } else if (!A->getOption().matches(ID: OPT_X__)) |
| 237 | continue; |
| 238 | |
| 239 | // Handle any user defined labels. |
| 240 | const StringRef Label = A->getValue(N: 0); |
| 241 | |
| 242 | // Ban "public" and "private" labels. |
| 243 | if ((Label.lower() == "public" ) || (Label.lower() == "private" )) { |
| 244 | Diags->Report(DiagID: diag::err_invalid_label) << Label; |
| 245 | return false; |
| 246 | } |
| 247 | |
| 248 | auto NextIt = std::next(x: It); |
| 249 | if (NextIt == End) { |
| 250 | Diags->Report(DiagID: clang::diag::err_drv_missing_argument) |
| 251 | << A->getAsString(Args) << 1; |
| 252 | return false; |
| 253 | } |
| 254 | Arg *NextA = *NextIt; |
| 255 | switch ((ID)NextA->getOption().getID()) { |
| 256 | case OPT_D: |
| 257 | case OPT_U: |
| 258 | break; |
| 259 | default: |
| 260 | Diags->Report(DiagID: clang::diag::err_drv_argument_not_allowed_with) |
| 261 | << A->getAsString(Args) << NextA->getAsString(Args); |
| 262 | return false; |
| 263 | } |
| 264 | const StringRef ASpelling = NextA->getSpelling(); |
| 265 | const auto &AValues = NextA->getValues(); |
| 266 | auto &UniqueArgs = FEOpts.UniqueArgs[Label]; |
| 267 | if (AValues.empty()) |
| 268 | UniqueArgs.emplace_back(args: ASpelling.str()); |
| 269 | else |
| 270 | for (const StringRef Val : AValues) |
| 271 | UniqueArgs.emplace_back(args: (ASpelling + Val).str()); |
| 272 | |
| 273 | A->claim(); |
| 274 | NextA->claim(); |
| 275 | } |
| 276 | |
| 277 | return true; |
| 278 | } |
| 279 | |
| 280 | bool Options::processXplatformOption(InputArgList &Args, arg_iterator Curr) { |
| 281 | Arg *A = *Curr; |
| 282 | |
| 283 | PlatformType Platform = getPlatformFromName(Name: A->getValue(N: 0)); |
| 284 | if (Platform == PLATFORM_UNKNOWN) { |
| 285 | Diags->Report(DiagID: diag::err_unsupported_os) |
| 286 | << getPlatformName(Platform) << A->getAsString(Args); |
| 287 | return false; |
| 288 | } |
| 289 | auto NextIt = std::next(x: Curr); |
| 290 | if (NextIt == Args.end()) { |
| 291 | Diags->Report(DiagID: diag::err_drv_missing_argument) << A->getAsString(Args) << 1; |
| 292 | return false; |
| 293 | } |
| 294 | |
| 295 | Arg *NextA = *NextIt; |
| 296 | switch ((ID)NextA->getOption().getID()) { |
| 297 | case OPT_iframework: |
| 298 | FEOpts.SystemFwkPaths.emplace_back(args: NextA->getValue(), args&: Platform); |
| 299 | break; |
| 300 | default: |
| 301 | Diags->Report(DiagID: diag::err_drv_invalid_argument_to_option) |
| 302 | << A->getAsString(Args) << NextA->getAsString(Args); |
| 303 | return false; |
| 304 | } |
| 305 | |
| 306 | A->claim(); |
| 307 | NextA->claim(); |
| 308 | |
| 309 | return true; |
| 310 | } |
| 311 | |
| 312 | bool Options::processXprojectOption(InputArgList &Args, arg_iterator Curr) { |
| 313 | Arg *A = *Curr; |
| 314 | auto NextIt = std::next(x: Curr); |
| 315 | if (NextIt == Args.end()) { |
| 316 | Diags->Report(DiagID: diag::err_drv_missing_argument) << A->getAsString(Args) << 1; |
| 317 | return false; |
| 318 | } |
| 319 | |
| 320 | Arg *NextA = *NextIt; |
| 321 | switch ((ID)NextA->getOption().getID()) { |
| 322 | case OPT_fobjc_arc: |
| 323 | case OPT_fmodules: |
| 324 | case OPT_fmodules_cache_path: |
| 325 | case OPT_include_: |
| 326 | case OPT_fvisibility_EQ: |
| 327 | break; |
| 328 | default: |
| 329 | Diags->Report(DiagID: diag::err_drv_argument_not_allowed_with) |
| 330 | << A->getAsString(Args) << NextA->getAsString(Args); |
| 331 | return false; |
| 332 | } |
| 333 | |
| 334 | std::string ArgString = NextA->getSpelling().str(); |
| 335 | for (const StringRef Val : NextA->getValues()) |
| 336 | ArgString += Val.str(); |
| 337 | |
| 338 | ProjectLevelArgs.push_back(x: ArgString); |
| 339 | A->claim(); |
| 340 | NextA->claim(); |
| 341 | |
| 342 | return true; |
| 343 | } |
| 344 | |
| 345 | bool Options::processXarchOption(InputArgList &Args, arg_iterator Curr) { |
| 346 | Arg *CurrArg = *Curr; |
| 347 | Architecture Arch = getArchitectureFromName(Name: CurrArg->getValue(N: 0)); |
| 348 | if (Arch == AK_unknown) { |
| 349 | Diags->Report(DiagID: diag::err_drv_invalid_arch_name) |
| 350 | << CurrArg->getAsString(Args); |
| 351 | return false; |
| 352 | } |
| 353 | |
| 354 | auto NextIt = std::next(x: Curr); |
| 355 | if (NextIt == Args.end()) { |
| 356 | Diags->Report(DiagID: diag::err_drv_missing_argument) |
| 357 | << CurrArg->getAsString(Args) << 1; |
| 358 | return false; |
| 359 | } |
| 360 | |
| 361 | // InstallAPI has a limited understanding of supported Xarch options. |
| 362 | // Currently this is restricted to linker inputs. |
| 363 | const Arg *NextArg = *NextIt; |
| 364 | switch (NextArg->getOption().getID()) { |
| 365 | case OPT_allowable_client: |
| 366 | case OPT_reexport_l: |
| 367 | case OPT_reexport_framework: |
| 368 | case OPT_reexport_library: |
| 369 | case OPT_rpath: |
| 370 | break; |
| 371 | default: |
| 372 | Diags->Report(DiagID: diag::err_drv_invalid_argument_to_option) |
| 373 | << NextArg->getAsString(Args) << CurrArg->getAsString(Args); |
| 374 | return false; |
| 375 | } |
| 376 | |
| 377 | ArgToArchMap[NextArg] = Arch; |
| 378 | CurrArg->claim(); |
| 379 | |
| 380 | return true; |
| 381 | } |
| 382 | |
| 383 | bool Options::processOptionList(InputArgList &Args, |
| 384 | llvm::opt::OptTable *Table) { |
| 385 | Arg *A = Args.getLastArg(Ids: OPT_option_list); |
| 386 | if (!A) |
| 387 | return true; |
| 388 | |
| 389 | const StringRef Path = A->getValue(N: 0); |
| 390 | auto InputOrErr = FM->getBufferForFile(Filename: Path); |
| 391 | if (auto Err = InputOrErr.getError()) { |
| 392 | Diags->Report(DiagID: diag::err_cannot_open_file) << Path << Err.message(); |
| 393 | return false; |
| 394 | } |
| 395 | // Backing storage referenced for argument processing. |
| 396 | std::vector<std::string> Storage; |
| 397 | auto ArgsOrErr = |
| 398 | getArgListFromJSON(Input: (*InputOrErr)->getBuffer(), Table, Storage); |
| 399 | |
| 400 | if (auto Err = ArgsOrErr.takeError()) { |
| 401 | Diags->Report(DiagID: diag::err_cannot_read_input_list) |
| 402 | << "option" << Path << toString(E: std::move(Err)); |
| 403 | return false; |
| 404 | } |
| 405 | return processInstallAPIXOptions(Args&: *ArgsOrErr); |
| 406 | } |
| 407 | |
| 408 | bool Options::processLinkerOptions(InputArgList &Args) { |
| 409 | // Handle required arguments. |
| 410 | if (const Arg *A = Args.getLastArg(Ids: drv::OPT_install__name)) |
| 411 | LinkerOpts.InstallName = A->getValue(); |
| 412 | if (LinkerOpts.InstallName.empty()) { |
| 413 | Diags->Report(DiagID: diag::err_no_install_name); |
| 414 | return false; |
| 415 | } |
| 416 | |
| 417 | // Defaulted or optional arguments. |
| 418 | if (auto *Arg = Args.getLastArg(Ids: drv::OPT_current__version)) |
| 419 | LinkerOpts.CurrentVersion.parse64(Str: Arg->getValue()); |
| 420 | |
| 421 | if (auto *Arg = Args.getLastArg(Ids: drv::OPT_compatibility__version)) |
| 422 | LinkerOpts.CompatVersion.parse64(Str: Arg->getValue()); |
| 423 | |
| 424 | if (auto *Arg = Args.getLastArg(Ids: drv::OPT_compatibility__version)) |
| 425 | LinkerOpts.CompatVersion.parse64(Str: Arg->getValue()); |
| 426 | |
| 427 | if (auto *Arg = Args.getLastArg(Ids: drv::OPT_umbrella)) |
| 428 | LinkerOpts.ParentUmbrella = Arg->getValue(); |
| 429 | |
| 430 | LinkerOpts.IsDylib = Args.hasArg(Ids: drv::OPT_dynamiclib); |
| 431 | |
| 432 | for (auto *Arg : Args.filtered(Ids: drv::OPT_alias_list)) { |
| 433 | LinkerOpts.AliasLists.emplace_back(args: Arg->getValue()); |
| 434 | Arg->claim(); |
| 435 | } |
| 436 | |
| 437 | LinkerOpts.AppExtensionSafe = Args.hasFlag( |
| 438 | Pos: drv::OPT_fapplication_extension, Neg: drv::OPT_fno_application_extension, |
| 439 | /*Default=*/LinkerOpts.AppExtensionSafe); |
| 440 | |
| 441 | if (::getenv(name: "LD_NO_ENCRYPT" ) != nullptr) |
| 442 | LinkerOpts.AppExtensionSafe = true; |
| 443 | |
| 444 | if (::getenv(name: "LD_APPLICATION_EXTENSION_SAFE" ) != nullptr) |
| 445 | LinkerOpts.AppExtensionSafe = true; |
| 446 | |
| 447 | // Capture library paths. |
| 448 | PathSeq LibraryPaths; |
| 449 | for (const Arg *A : Args.filtered(Ids: drv::OPT_L)) { |
| 450 | LibraryPaths.emplace_back(args: A->getValue()); |
| 451 | A->claim(); |
| 452 | } |
| 453 | |
| 454 | if (!LibraryPaths.empty()) |
| 455 | LinkerOpts.LibPaths = std::move(LibraryPaths); |
| 456 | |
| 457 | return true; |
| 458 | } |
| 459 | |
| 460 | // NOTE: Do not claim any arguments, as they will be passed along for CC1 |
| 461 | // invocations. |
| 462 | bool Options::processFrontendOptions(InputArgList &Args) { |
| 463 | // Capture language mode. |
| 464 | if (auto *A = Args.getLastArgNoClaim(Ids: drv::OPT_x)) { |
| 465 | FEOpts.LangMode = llvm::StringSwitch<clang::Language>(A->getValue()) |
| 466 | .Case(S: "c" , Value: clang::Language::C) |
| 467 | .Case(S: "c++" , Value: clang::Language::CXX) |
| 468 | .Case(S: "objective-c" , Value: clang::Language::ObjC) |
| 469 | .Case(S: "objective-c++" , Value: clang::Language::ObjCXX) |
| 470 | .Default(Value: clang::Language::Unknown); |
| 471 | |
| 472 | if (FEOpts.LangMode == clang::Language::Unknown) { |
| 473 | Diags->Report(DiagID: clang::diag::err_drv_invalid_value) |
| 474 | << A->getAsString(Args) << A->getValue(); |
| 475 | return false; |
| 476 | } |
| 477 | } |
| 478 | for (auto *A : Args.filtered(Ids: drv::OPT_ObjC, Ids: drv::OPT_ObjCXX)) { |
| 479 | if (A->getOption().matches(ID: drv::OPT_ObjC)) |
| 480 | FEOpts.LangMode = clang::Language::ObjC; |
| 481 | else |
| 482 | FEOpts.LangMode = clang::Language::ObjCXX; |
| 483 | } |
| 484 | |
| 485 | // Capture Sysroot. |
| 486 | if (const Arg *A = Args.getLastArgNoClaim(Ids: drv::OPT_isysroot)) { |
| 487 | SmallString<PATH_MAX> Path(A->getValue()); |
| 488 | FM->makeAbsolutePath(Path); |
| 489 | if (!FM->getOptionalDirectoryRef(DirName: Path)) { |
| 490 | Diags->Report(DiagID: diag::err_missing_sysroot) << Path; |
| 491 | return false; |
| 492 | } |
| 493 | FEOpts.ISysroot = std::string(Path); |
| 494 | } else if (FEOpts.ISysroot.empty()) { |
| 495 | // Mirror CLANG and obtain the isysroot from the SDKROOT environment |
| 496 | // variable, if it wasn't defined by the command line. |
| 497 | if (auto *Env = ::getenv(name: "SDKROOT" )) { |
| 498 | if (StringRef(Env) != "/" && llvm::sys::path::is_absolute(path: Env) && |
| 499 | FM->getOptionalFileRef(Filename: Env)) |
| 500 | FEOpts.ISysroot = Env; |
| 501 | } |
| 502 | } |
| 503 | |
| 504 | // Capture system frameworks for all platforms. |
| 505 | for (const Arg *A : Args.filtered(Ids: drv::OPT_iframework)) |
| 506 | FEOpts.SystemFwkPaths.emplace_back(args: A->getValue(), |
| 507 | args: std::optional<PlatformType>{}); |
| 508 | |
| 509 | // Capture framework paths. |
| 510 | PathSeq FrameworkPaths; |
| 511 | for (const Arg *A : Args.filtered(Ids: drv::OPT_F)) |
| 512 | FrameworkPaths.emplace_back(args: A->getValue()); |
| 513 | |
| 514 | if (!FrameworkPaths.empty()) |
| 515 | FEOpts.FwkPaths = std::move(FrameworkPaths); |
| 516 | |
| 517 | // Add default framework/library paths. |
| 518 | PathSeq DefaultLibraryPaths = {"/usr/lib" , "/usr/local/lib" }; |
| 519 | PathSeq DefaultFrameworkPaths = {"/Library/Frameworks" , |
| 520 | "/System/Library/Frameworks" }; |
| 521 | |
| 522 | for (const StringRef LibPath : DefaultLibraryPaths) { |
| 523 | SmallString<PATH_MAX> Path(FEOpts.ISysroot); |
| 524 | sys::path::append(path&: Path, a: LibPath); |
| 525 | LinkerOpts.LibPaths.emplace_back(args: Path.str()); |
| 526 | } |
| 527 | for (const StringRef FwkPath : DefaultFrameworkPaths) { |
| 528 | SmallString<PATH_MAX> Path(FEOpts.ISysroot); |
| 529 | sys::path::append(path&: Path, a: FwkPath); |
| 530 | FEOpts.SystemFwkPaths.emplace_back(args: Path.str(), |
| 531 | args: std::optional<PlatformType>{}); |
| 532 | } |
| 533 | |
| 534 | return true; |
| 535 | } |
| 536 | |
| 537 | bool Options::addFilePaths(InputArgList &Args, PathSeq &, |
| 538 | OptSpecifier ID) { |
| 539 | for (const StringRef Path : Args.getAllArgValues(Id: ID)) { |
| 540 | if ((bool)FM->getOptionalDirectoryRef(DirName: Path, /*CacheFailure=*/false)) { |
| 541 | auto = enumerateFiles(FM&: *FM, Directory: Path); |
| 542 | if (!InputHeadersOrErr) { |
| 543 | Diags->Report(DiagID: diag::err_cannot_open_file) |
| 544 | << Path << toString(E: InputHeadersOrErr.takeError()); |
| 545 | return false; |
| 546 | } |
| 547 | // Sort headers to ensure deterministic behavior. |
| 548 | sort(C&: *InputHeadersOrErr); |
| 549 | for (StringRef H : *InputHeadersOrErr) |
| 550 | Headers.emplace_back(args: std::move(H)); |
| 551 | } else |
| 552 | Headers.emplace_back(args: Path); |
| 553 | } |
| 554 | return true; |
| 555 | } |
| 556 | |
| 557 | std::vector<const char *> |
| 558 | Options::processAndFilterOutInstallAPIOptions(ArrayRef<const char *> Args) { |
| 559 | std::unique_ptr<llvm::opt::OptTable> Table; |
| 560 | Table.reset(p: createDriverOptTable()); |
| 561 | |
| 562 | unsigned MissingArgIndex, MissingArgCount; |
| 563 | auto ParsedArgs = Table->ParseArgs(Args: Args.slice(N: 1), MissingArgIndex, |
| 564 | MissingArgCount, FlagsToInclude: Visibility()); |
| 565 | |
| 566 | // Capture InstallAPI only driver options. |
| 567 | if (!processInstallAPIXOptions(Args&: ParsedArgs)) |
| 568 | return {}; |
| 569 | |
| 570 | if (!processOptionList(Args&: ParsedArgs, Table: Table.get())) |
| 571 | return {}; |
| 572 | |
| 573 | DriverOpts.Demangle = ParsedArgs.hasArg(Ids: OPT_demangle); |
| 574 | |
| 575 | if (auto *A = ParsedArgs.getLastArg(Ids: OPT_filetype)) { |
| 576 | DriverOpts.OutFT = TextAPIWriter::parseFileType(FT: A->getValue()); |
| 577 | if (DriverOpts.OutFT == FileType::Invalid) { |
| 578 | Diags->Report(DiagID: clang::diag::err_drv_invalid_value) |
| 579 | << A->getAsString(Args: ParsedArgs) << A->getValue(); |
| 580 | return {}; |
| 581 | } |
| 582 | } |
| 583 | |
| 584 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_verify_mode_EQ)) { |
| 585 | DriverOpts.VerifyMode = |
| 586 | StringSwitch<VerificationMode>(A->getValue()) |
| 587 | .Case(S: "ErrorsOnly" , Value: VerificationMode::ErrorsOnly) |
| 588 | .Case(S: "ErrorsAndWarnings" , Value: VerificationMode::ErrorsAndWarnings) |
| 589 | .Case(S: "Pedantic" , Value: VerificationMode::Pedantic) |
| 590 | .Default(Value: VerificationMode::Invalid); |
| 591 | |
| 592 | if (DriverOpts.VerifyMode == VerificationMode::Invalid) { |
| 593 | Diags->Report(DiagID: clang::diag::err_drv_invalid_value) |
| 594 | << A->getAsString(Args: ParsedArgs) << A->getValue(); |
| 595 | return {}; |
| 596 | } |
| 597 | } |
| 598 | |
| 599 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_verify_against)) |
| 600 | DriverOpts.DylibToVerify = A->getValue(); |
| 601 | |
| 602 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_dsym)) |
| 603 | DriverOpts.DSYMPath = A->getValue(); |
| 604 | |
| 605 | DriverOpts.TraceLibraryLocation = ParsedArgs.hasArg(Ids: OPT_t); |
| 606 | |
| 607 | // Linker options not handled by clang driver. |
| 608 | LinkerOpts.OSLibNotForSharedCache = |
| 609 | ParsedArgs.hasArg(Ids: OPT_not_for_dyld_shared_cache); |
| 610 | |
| 611 | for (const Arg *A : ParsedArgs.filtered(Ids: OPT_allowable_client)) { |
| 612 | auto It = ArgToArchMap.find(Val: A); |
| 613 | LinkerOpts.AllowableClients.getArchSet(Attr: A->getValue()) = |
| 614 | It != ArgToArchMap.end() ? It->second : ArchitectureSet(); |
| 615 | A->claim(); |
| 616 | } |
| 617 | |
| 618 | for (const Arg *A : ParsedArgs.filtered(Ids: OPT_reexport_l)) { |
| 619 | auto It = ArgToArchMap.find(Val: A); |
| 620 | LinkerOpts.ReexportedLibraries.getArchSet(Attr: A->getValue()) = |
| 621 | It != ArgToArchMap.end() ? It->second : ArchitectureSet(); |
| 622 | A->claim(); |
| 623 | } |
| 624 | |
| 625 | for (const Arg *A : ParsedArgs.filtered(Ids: OPT_reexport_library)) { |
| 626 | auto It = ArgToArchMap.find(Val: A); |
| 627 | LinkerOpts.ReexportedLibraryPaths.getArchSet(Attr: A->getValue()) = |
| 628 | It != ArgToArchMap.end() ? It->second : ArchitectureSet(); |
| 629 | A->claim(); |
| 630 | } |
| 631 | |
| 632 | for (const Arg *A : ParsedArgs.filtered(Ids: OPT_reexport_framework)) { |
| 633 | auto It = ArgToArchMap.find(Val: A); |
| 634 | LinkerOpts.ReexportedFrameworks.getArchSet(Attr: A->getValue()) = |
| 635 | It != ArgToArchMap.end() ? It->second : ArchitectureSet(); |
| 636 | A->claim(); |
| 637 | } |
| 638 | |
| 639 | for (const Arg *A : ParsedArgs.filtered(Ids: OPT_rpath)) { |
| 640 | auto It = ArgToArchMap.find(Val: A); |
| 641 | LinkerOpts.RPaths.getArchSet(Attr: A->getValue()) = |
| 642 | It != ArgToArchMap.end() ? It->second : ArchitectureSet(); |
| 643 | A->claim(); |
| 644 | } |
| 645 | |
| 646 | // Handle exclude & extra header directories or files. |
| 647 | auto handleAdditionalInputArgs = [&](PathSeq &, |
| 648 | clang::installapi::ID OptID) { |
| 649 | if (ParsedArgs.hasArgNoClaim(Ids: OptID)) |
| 650 | Headers.clear(); |
| 651 | return addFilePaths(Args&: ParsedArgs, Headers, ID: OptID); |
| 652 | }; |
| 653 | |
| 654 | if (!handleAdditionalInputArgs(DriverOpts.ExtraPublicHeaders, |
| 655 | OPT_extra_public_header)) |
| 656 | return {}; |
| 657 | |
| 658 | if (!handleAdditionalInputArgs(DriverOpts.ExtraPrivateHeaders, |
| 659 | OPT_extra_private_header)) |
| 660 | return {}; |
| 661 | if (!handleAdditionalInputArgs(DriverOpts.ExtraProjectHeaders, |
| 662 | OPT_extra_project_header)) |
| 663 | return {}; |
| 664 | |
| 665 | if (!handleAdditionalInputArgs(DriverOpts.ExcludePublicHeaders, |
| 666 | OPT_exclude_public_header)) |
| 667 | return {}; |
| 668 | if (!handleAdditionalInputArgs(DriverOpts.ExcludePrivateHeaders, |
| 669 | OPT_exclude_private_header)) |
| 670 | return {}; |
| 671 | if (!handleAdditionalInputArgs(DriverOpts.ExcludeProjectHeaders, |
| 672 | OPT_exclude_project_header)) |
| 673 | return {}; |
| 674 | |
| 675 | // Handle umbrella headers. |
| 676 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_public_umbrella_header)) |
| 677 | DriverOpts.PublicUmbrellaHeader = A->getValue(); |
| 678 | |
| 679 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_private_umbrella_header)) |
| 680 | DriverOpts.PrivateUmbrellaHeader = A->getValue(); |
| 681 | |
| 682 | if (const Arg *A = ParsedArgs.getLastArg(Ids: OPT_project_umbrella_header)) |
| 683 | DriverOpts.ProjectUmbrellaHeader = A->getValue(); |
| 684 | |
| 685 | /// Any unclaimed arguments should be forwarded to the clang driver. |
| 686 | std::vector<const char *> ClangDriverArgs(ParsedArgs.size()); |
| 687 | for (const Arg *A : ParsedArgs) { |
| 688 | if (A->isClaimed()) |
| 689 | continue; |
| 690 | // Forward along unclaimed but overlapping arguments to the clang driver. |
| 691 | if (A->getOption().getID() > (unsigned)OPT_UNKNOWN) { |
| 692 | ClangDriverArgs.push_back(x: A->getSpelling().data()); |
| 693 | } else |
| 694 | llvm::append_range(C&: ClangDriverArgs, R: A->getValues()); |
| 695 | } |
| 696 | return ClangDriverArgs; |
| 697 | } |
| 698 | |
| 699 | Options::Options(DiagnosticsEngine &Diag, FileManager *FM, |
| 700 | ArrayRef<const char *> Args, const StringRef ProgName) |
| 701 | : Diags(&Diag), FM(FM) { |
| 702 | |
| 703 | // First process InstallAPI specific options. |
| 704 | auto DriverArgs = processAndFilterOutInstallAPIOptions(Args); |
| 705 | if (Diags->hasErrorOccurred()) |
| 706 | return; |
| 707 | |
| 708 | // Set up driver to parse remaining input arguments. |
| 709 | clang::driver::Driver Driver(ProgName, llvm::sys::getDefaultTargetTriple(), |
| 710 | *Diags, "clang installapi tool" ); |
| 711 | auto TargetAndMode = |
| 712 | clang::driver::ToolChain::getTargetAndModeFromProgramName(ProgName); |
| 713 | Driver.setTargetAndMode(TargetAndMode); |
| 714 | bool HasError = false; |
| 715 | llvm::opt::InputArgList ArgList = |
| 716 | Driver.ParseArgStrings(Args: DriverArgs, /*UseDriverMode=*/true, ContainsError&: HasError); |
| 717 | if (HasError) |
| 718 | return; |
| 719 | Driver.setCheckInputsExist(false); |
| 720 | |
| 721 | if (!processDriverOptions(Args&: ArgList)) |
| 722 | return; |
| 723 | |
| 724 | if (!processLinkerOptions(Args&: ArgList)) |
| 725 | return; |
| 726 | |
| 727 | if (!processFrontendOptions(Args&: ArgList)) |
| 728 | return; |
| 729 | |
| 730 | // After all InstallAPI necessary arguments have been collected. Go back and |
| 731 | // assign values that were unknown before the clang driver opt table was used. |
| 732 | ArchitectureSet AllArchs; |
| 733 | for (const auto &T : DriverOpts.Targets) |
| 734 | AllArchs.set(T.first.Arch); |
| 735 | auto assignDefaultLibAttrs = [&AllArchs](LibAttrs &Attrs) { |
| 736 | for (auto &[_, Archs] : Attrs.get()) |
| 737 | if (Archs.empty()) |
| 738 | Archs = AllArchs; |
| 739 | }; |
| 740 | assignDefaultLibAttrs(LinkerOpts.AllowableClients); |
| 741 | assignDefaultLibAttrs(LinkerOpts.ReexportedFrameworks); |
| 742 | assignDefaultLibAttrs(LinkerOpts.ReexportedLibraries); |
| 743 | assignDefaultLibAttrs(LinkerOpts.ReexportedLibraryPaths); |
| 744 | assignDefaultLibAttrs(LinkerOpts.RPaths); |
| 745 | |
| 746 | /// Force cc1 options that should always be on. |
| 747 | FrontendArgs = {"-fsyntax-only" , "-Wprivate-extern" }; |
| 748 | |
| 749 | /// Any unclaimed arguments should be handled by invoking the clang frontend. |
| 750 | for (const Arg *A : ArgList) { |
| 751 | if (A->isClaimed()) |
| 752 | continue; |
| 753 | FrontendArgs.emplace_back(args: A->getSpelling()); |
| 754 | llvm::append_range(C&: FrontendArgs, R: A->getValues()); |
| 755 | } |
| 756 | } |
| 757 | |
| 758 | static Expected<std::unique_ptr<InterfaceFile>> |
| 759 | getInterfaceFile(const StringRef Filename) { |
| 760 | ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr = |
| 761 | MemoryBuffer::getFile(Filename); |
| 762 | if (auto Err = BufferOrErr.getError()) |
| 763 | return errorCodeToError(EC: std::move(Err)); |
| 764 | |
| 765 | auto Buffer = std::move(*BufferOrErr); |
| 766 | switch (identify_magic(magic: Buffer->getBuffer())) { |
| 767 | case file_magic::macho_dynamically_linked_shared_lib: |
| 768 | case file_magic::macho_dynamically_linked_shared_lib_stub: |
| 769 | case file_magic::macho_universal_binary: |
| 770 | return DylibReader::get(Buffer: Buffer->getMemBufferRef()); |
| 771 | break; |
| 772 | case file_magic::tapi_file: |
| 773 | return TextAPIReader::get(InputBuffer: Buffer->getMemBufferRef()); |
| 774 | default: |
| 775 | return make_error<TextAPIError>(Args: TextAPIErrorCode::InvalidInputFormat, |
| 776 | Args: "unsupported library file format" ); |
| 777 | } |
| 778 | llvm_unreachable("unexpected failure in getInterface" ); |
| 779 | } |
| 780 | |
| 781 | std::pair<LibAttrs, ReexportedInterfaces> Options::getReexportedLibraries() { |
| 782 | LibAttrs Reexports; |
| 783 | ReexportedInterfaces ReexportIFs; |
| 784 | auto AccumulateReexports = [&](StringRef Path, const ArchitectureSet &Archs) { |
| 785 | auto ReexportIFOrErr = getInterfaceFile(Filename: Path); |
| 786 | if (!ReexportIFOrErr) |
| 787 | return false; |
| 788 | std::unique_ptr<InterfaceFile> Reexport = std::move(*ReexportIFOrErr); |
| 789 | StringRef InstallName = Reexport->getInstallName(); |
| 790 | assert(!InstallName.empty() && "Parse error for install name" ); |
| 791 | Reexports.getArchSet(Attr: InstallName) = Archs; |
| 792 | ReexportIFs.emplace_back(Args: std::move(*Reexport)); |
| 793 | return true; |
| 794 | }; |
| 795 | |
| 796 | PlatformSet Platforms; |
| 797 | for (const auto &T : DriverOpts.Targets) |
| 798 | Platforms.insert(V: T.first.Platform); |
| 799 | // Populate search paths by looking at user paths before system ones. |
| 800 | PathSeq FwkSearchPaths(FEOpts.FwkPaths.begin(), FEOpts.FwkPaths.end()); |
| 801 | for (const PlatformType P : Platforms) { |
| 802 | PathSeq PlatformSearchPaths = getPathsForPlatform(Paths: FEOpts.SystemFwkPaths, Platform: P); |
| 803 | llvm::append_range(C&: FwkSearchPaths, R&: PlatformSearchPaths); |
| 804 | for (const auto &[Lib, Archs] : LinkerOpts.ReexportedFrameworks.get()) { |
| 805 | std::string Name = (Lib + ".framework/" + Lib); |
| 806 | std::string Path = findLibrary(InstallName: Name, FM&: *FM, FrameworkSearchPaths: FwkSearchPaths, LibrarySearchPaths: {}, SearchPaths: {}); |
| 807 | if (Path.empty()) { |
| 808 | Diags->Report(DiagID: diag::err_cannot_find_reexport) << false << Lib; |
| 809 | return {}; |
| 810 | } |
| 811 | if (DriverOpts.TraceLibraryLocation) |
| 812 | errs() << Path << "\n" ; |
| 813 | |
| 814 | AccumulateReexports(Path, Archs); |
| 815 | } |
| 816 | FwkSearchPaths.resize(new_size: FwkSearchPaths.size() - PlatformSearchPaths.size()); |
| 817 | } |
| 818 | |
| 819 | for (const auto &[Lib, Archs] : LinkerOpts.ReexportedLibraries.get()) { |
| 820 | std::string Name = "lib" + Lib + ".dylib" ; |
| 821 | std::string Path = findLibrary(InstallName: Name, FM&: *FM, FrameworkSearchPaths: {}, LibrarySearchPaths: LinkerOpts.LibPaths, SearchPaths: {}); |
| 822 | if (Path.empty()) { |
| 823 | Diags->Report(DiagID: diag::err_cannot_find_reexport) << true << Lib; |
| 824 | return {}; |
| 825 | } |
| 826 | if (DriverOpts.TraceLibraryLocation) |
| 827 | errs() << Path << "\n" ; |
| 828 | |
| 829 | AccumulateReexports(Path, Archs); |
| 830 | } |
| 831 | |
| 832 | for (const auto &[Lib, Archs] : LinkerOpts.ReexportedLibraryPaths.get()) |
| 833 | AccumulateReexports(Lib, Archs); |
| 834 | |
| 835 | return {std::move(Reexports), std::move(ReexportIFs)}; |
| 836 | } |
| 837 | |
| 838 | InstallAPIContext Options::createContext() { |
| 839 | InstallAPIContext Ctx; |
| 840 | Ctx.FM = FM; |
| 841 | Ctx.Diags = Diags; |
| 842 | |
| 843 | // InstallAPI requires two level namespacing. |
| 844 | Ctx.BA.TwoLevelNamespace = true; |
| 845 | |
| 846 | Ctx.BA.InstallName = LinkerOpts.InstallName; |
| 847 | Ctx.BA.CurrentVersion = LinkerOpts.CurrentVersion; |
| 848 | Ctx.BA.CompatVersion = LinkerOpts.CompatVersion; |
| 849 | Ctx.BA.AppExtensionSafe = LinkerOpts.AppExtensionSafe; |
| 850 | Ctx.BA.ParentUmbrella = LinkerOpts.ParentUmbrella; |
| 851 | Ctx.BA.OSLibNotForSharedCache = LinkerOpts.OSLibNotForSharedCache; |
| 852 | Ctx.FT = DriverOpts.OutFT; |
| 853 | Ctx.OutputLoc = DriverOpts.OutputPath; |
| 854 | Ctx.LangMode = FEOpts.LangMode; |
| 855 | |
| 856 | auto [Reexports, ReexportedIFs] = getReexportedLibraries(); |
| 857 | if (Diags->hasErrorOccurred()) |
| 858 | return Ctx; |
| 859 | Ctx.Reexports = Reexports; |
| 860 | |
| 861 | // Collect symbols from alias lists. |
| 862 | AliasMap Aliases; |
| 863 | for (const StringRef ListPath : LinkerOpts.AliasLists) { |
| 864 | auto Buffer = FM->getBufferForFile(Filename: ListPath); |
| 865 | if (auto Err = Buffer.getError()) { |
| 866 | Diags->Report(DiagID: diag::err_cannot_open_file) << ListPath << Err.message(); |
| 867 | return Ctx; |
| 868 | } |
| 869 | Expected<AliasMap> Result = parseAliasList(Buffer&: Buffer.get()); |
| 870 | if (!Result) { |
| 871 | Diags->Report(DiagID: diag::err_cannot_read_input_list) |
| 872 | << "symbol alias" << ListPath << toString(E: Result.takeError()); |
| 873 | return Ctx; |
| 874 | } |
| 875 | Aliases.insert(first: Result.get().begin(), last: Result.get().end()); |
| 876 | } |
| 877 | |
| 878 | // Attempt to find umbrella headers by capturing framework name. |
| 879 | StringRef FrameworkName; |
| 880 | if (!LinkerOpts.IsDylib) |
| 881 | FrameworkName = |
| 882 | Library::getFrameworkNameFromInstallName(InstallName: LinkerOpts.InstallName); |
| 883 | |
| 884 | /// Process inputs headers. |
| 885 | // 1. For headers discovered by directory scanning, sort them. |
| 886 | // 2. For headers discovered by filelist, respect ordering. |
| 887 | // 3. Append extra headers and mark any excluded headers. |
| 888 | // 4. Finally, surface up umbrella headers to top of the list. |
| 889 | if (!DriverOpts.InputDirectory.empty()) { |
| 890 | DirectoryScanner Scanner(*FM, LinkerOpts.IsDylib |
| 891 | ? ScanMode::ScanDylibs |
| 892 | : ScanMode::ScanFrameworks); |
| 893 | SmallString<PATH_MAX> NormalizedPath(DriverOpts.InputDirectory); |
| 894 | FM->getVirtualFileSystem().makeAbsolute(Path&: NormalizedPath); |
| 895 | sys::path::remove_dots(path&: NormalizedPath, /*remove_dot_dot=*/true); |
| 896 | if (llvm::Error Err = Scanner.scan(Directory: NormalizedPath)) { |
| 897 | Diags->Report(DiagID: diag::err_directory_scanning) |
| 898 | << DriverOpts.InputDirectory << std::move(Err); |
| 899 | return Ctx; |
| 900 | } |
| 901 | std::vector<Library> InputLibraries = Scanner.takeLibraries(); |
| 902 | if (InputLibraries.size() > 1) { |
| 903 | Diags->Report(DiagID: diag::err_more_than_one_library); |
| 904 | return Ctx; |
| 905 | } |
| 906 | llvm::append_range(C&: Ctx.InputHeaders, |
| 907 | R: DirectoryScanner::getHeaders(Libraries: InputLibraries)); |
| 908 | llvm::stable_sort(Range&: Ctx.InputHeaders); |
| 909 | } |
| 910 | |
| 911 | for (const StringRef ListPath : DriverOpts.FileLists) { |
| 912 | auto Buffer = FM->getBufferForFile(Filename: ListPath); |
| 913 | if (auto Err = Buffer.getError()) { |
| 914 | Diags->Report(DiagID: diag::err_cannot_open_file) << ListPath << Err.message(); |
| 915 | return Ctx; |
| 916 | } |
| 917 | if (auto Err = FileListReader::loadHeaders(InputBuffer: std::move(Buffer.get()), |
| 918 | Destination&: Ctx.InputHeaders, FM)) { |
| 919 | Diags->Report(DiagID: diag::err_cannot_read_input_list) |
| 920 | << "header file" << ListPath << std::move(Err); |
| 921 | return Ctx; |
| 922 | } |
| 923 | } |
| 924 | // After initial input has been processed, add any extra headers. |
| 925 | auto HandleExtraHeaders = [&](PathSeq &, HeaderType Type) -> bool { |
| 926 | assert(Type != HeaderType::Unknown && "Missing header type." ); |
| 927 | for (const StringRef Path : Headers) { |
| 928 | if (!FM->getOptionalFileRef(Filename: Path)) { |
| 929 | Diags->Report(DiagID: diag::err_no_such_header_file) << Path << (unsigned)Type; |
| 930 | return false; |
| 931 | } |
| 932 | SmallString<PATH_MAX> FullPath(Path); |
| 933 | FM->makeAbsolutePath(Path&: FullPath); |
| 934 | |
| 935 | auto IncludeName = createIncludeHeaderName(FullPath); |
| 936 | Ctx.InputHeaders.emplace_back( |
| 937 | args&: FullPath, args&: Type, args: IncludeName.has_value() ? *IncludeName : "" ); |
| 938 | Ctx.InputHeaders.back().setExtra(); |
| 939 | } |
| 940 | return true; |
| 941 | }; |
| 942 | |
| 943 | if (!HandleExtraHeaders(DriverOpts.ExtraPublicHeaders, HeaderType::Public) || |
| 944 | !HandleExtraHeaders(DriverOpts.ExtraPrivateHeaders, |
| 945 | HeaderType::Private) || |
| 946 | !HandleExtraHeaders(DriverOpts.ExtraProjectHeaders, HeaderType::Project)) |
| 947 | return Ctx; |
| 948 | |
| 949 | // After all headers have been added, consider excluded headers. |
| 950 | std::vector<std::unique_ptr<HeaderGlob>> ; |
| 951 | std::set<FileEntryRef> ; |
| 952 | auto ParseGlobs = [&](const PathSeq &Paths, HeaderType Type) { |
| 953 | assert(Type != HeaderType::Unknown && "Missing header type." ); |
| 954 | for (const StringRef Path : Paths) { |
| 955 | auto Glob = HeaderGlob::create(GlobString: Path, Type); |
| 956 | if (Glob) |
| 957 | ExcludedHeaderGlobs.emplace_back(args: std::move(Glob.get())); |
| 958 | else { |
| 959 | consumeError(Err: Glob.takeError()); |
| 960 | if (auto File = FM->getFileRef(Filename: Path)) |
| 961 | ExcludedHeaderFiles.emplace(args&: *File); |
| 962 | else { |
| 963 | Diags->Report(DiagID: diag::err_no_such_header_file) |
| 964 | << Path << (unsigned)Type; |
| 965 | return false; |
| 966 | } |
| 967 | } |
| 968 | } |
| 969 | return true; |
| 970 | }; |
| 971 | |
| 972 | if (!ParseGlobs(DriverOpts.ExcludePublicHeaders, HeaderType::Public) || |
| 973 | !ParseGlobs(DriverOpts.ExcludePrivateHeaders, HeaderType::Private) || |
| 974 | !ParseGlobs(DriverOpts.ExcludeProjectHeaders, HeaderType::Project)) |
| 975 | return Ctx; |
| 976 | |
| 977 | for (HeaderFile & : Ctx.InputHeaders) { |
| 978 | for (auto &Glob : ExcludedHeaderGlobs) |
| 979 | if (Glob->match(Header)) |
| 980 | Header.setExcluded(); |
| 981 | } |
| 982 | if (!ExcludedHeaderFiles.empty()) { |
| 983 | for (HeaderFile & : Ctx.InputHeaders) { |
| 984 | auto FileRef = FM->getFileRef(Filename: Header.getPath()); |
| 985 | if (!FileRef) |
| 986 | continue; |
| 987 | if (ExcludedHeaderFiles.count(x: *FileRef)) |
| 988 | Header.setExcluded(); |
| 989 | } |
| 990 | } |
| 991 | // Report if glob was ignored. |
| 992 | for (const auto &Glob : ExcludedHeaderGlobs) |
| 993 | if (!Glob->didMatch()) |
| 994 | Diags->Report(DiagID: diag::warn_glob_did_not_match) << Glob->str(); |
| 995 | |
| 996 | // Mark any explicit or inferred umbrella headers. If one exists, move |
| 997 | // that to the beginning of the input headers. |
| 998 | auto MarkandMoveUmbrellaInHeaders = [&](llvm::Regex &Regex, |
| 999 | HeaderType Type) -> bool { |
| 1000 | auto It = find_if(Range&: Ctx.InputHeaders, P: [&Regex, Type](const HeaderFile &H) { |
| 1001 | return (H.getType() == Type) && Regex.match(String: H.getPath()); |
| 1002 | }); |
| 1003 | |
| 1004 | if (It == Ctx.InputHeaders.end()) |
| 1005 | return false; |
| 1006 | It->setUmbrellaHeader(); |
| 1007 | |
| 1008 | // Because there can be an umbrella header per header type, |
| 1009 | // find the first non umbrella header to swap position with. |
| 1010 | auto BeginPos = find_if(Range&: Ctx.InputHeaders, P: [](const HeaderFile &H) { |
| 1011 | return !H.isUmbrellaHeader(); |
| 1012 | }); |
| 1013 | if (BeginPos != Ctx.InputHeaders.end() && BeginPos < It) |
| 1014 | std::swap(a&: *BeginPos, b&: *It); |
| 1015 | return true; |
| 1016 | }; |
| 1017 | |
| 1018 | auto = [&](StringRef , HeaderType Type) -> bool { |
| 1019 | assert(Type != HeaderType::Unknown && "Missing header type." ); |
| 1020 | if (!HeaderPath.empty()) { |
| 1021 | auto EscapedString = Regex::escape(String: HeaderPath); |
| 1022 | Regex UmbrellaRegex(EscapedString); |
| 1023 | if (!MarkandMoveUmbrellaInHeaders(UmbrellaRegex, Type)) { |
| 1024 | Diags->Report(DiagID: diag::err_no_such_umbrella_header_file) |
| 1025 | << HeaderPath << (unsigned)Type; |
| 1026 | return false; |
| 1027 | } |
| 1028 | } else if (!FrameworkName.empty() && (Type != HeaderType::Project)) { |
| 1029 | auto UmbrellaName = "/" + Regex::escape(String: FrameworkName); |
| 1030 | if (Type == HeaderType::Public) |
| 1031 | UmbrellaName += "\\.h" ; |
| 1032 | else |
| 1033 | UmbrellaName += "[_]?Private\\.h" ; |
| 1034 | Regex UmbrellaRegex(UmbrellaName); |
| 1035 | MarkandMoveUmbrellaInHeaders(UmbrellaRegex, Type); |
| 1036 | } |
| 1037 | return true; |
| 1038 | }; |
| 1039 | if (!FindUmbrellaHeader(DriverOpts.PublicUmbrellaHeader, |
| 1040 | HeaderType::Public) || |
| 1041 | !FindUmbrellaHeader(DriverOpts.PrivateUmbrellaHeader, |
| 1042 | HeaderType::Private) || |
| 1043 | !FindUmbrellaHeader(DriverOpts.ProjectUmbrellaHeader, |
| 1044 | HeaderType::Project)) |
| 1045 | return Ctx; |
| 1046 | |
| 1047 | // Parse binary dylib and initialize verifier. |
| 1048 | if (DriverOpts.DylibToVerify.empty()) { |
| 1049 | Ctx.Verifier = std::make_unique<DylibVerifier>(); |
| 1050 | return Ctx; |
| 1051 | } |
| 1052 | |
| 1053 | auto Buffer = FM->getBufferForFile(Filename: DriverOpts.DylibToVerify); |
| 1054 | if (auto Err = Buffer.getError()) { |
| 1055 | Diags->Report(DiagID: diag::err_cannot_open_file) |
| 1056 | << DriverOpts.DylibToVerify << Err.message(); |
| 1057 | return Ctx; |
| 1058 | } |
| 1059 | |
| 1060 | DylibReader::ParseOption PO; |
| 1061 | PO.Undefineds = false; |
| 1062 | Expected<Records> Slices = |
| 1063 | DylibReader::readFile(Buffer: (*Buffer)->getMemBufferRef(), Opt: PO); |
| 1064 | if (auto Err = Slices.takeError()) { |
| 1065 | Diags->Report(DiagID: diag::err_cannot_open_file) |
| 1066 | << DriverOpts.DylibToVerify << std::move(Err); |
| 1067 | return Ctx; |
| 1068 | } |
| 1069 | |
| 1070 | Ctx.Verifier = std::make_unique<DylibVerifier>( |
| 1071 | args: std::move(*Slices), args: std::move(ReexportedIFs), args: std::move(Aliases), args&: Diags, |
| 1072 | args&: DriverOpts.VerifyMode, args&: DriverOpts.Zippered, args&: DriverOpts.Demangle, |
| 1073 | args&: DriverOpts.DSYMPath); |
| 1074 | return Ctx; |
| 1075 | } |
| 1076 | |
| 1077 | void Options::(std::vector<std::string> &ArgStrings, |
| 1078 | const llvm::Triple &Targ, |
| 1079 | const HeaderType Type) { |
| 1080 | // Unique to architecture (Xarch) options hold no arguments to pass along for |
| 1081 | // frontend. |
| 1082 | |
| 1083 | // Add specific to platform arguments. |
| 1084 | PathSeq PlatformSearchPaths = |
| 1085 | getPathsForPlatform(Paths: FEOpts.SystemFwkPaths, Platform: mapToPlatformType(Target: Targ)); |
| 1086 | for (StringRef Path : PlatformSearchPaths) { |
| 1087 | ArgStrings.push_back(x: "-iframework" ); |
| 1088 | ArgStrings.push_back(x: Path.str()); |
| 1089 | } |
| 1090 | |
| 1091 | // Add specific to header type arguments. |
| 1092 | if (Type == HeaderType::Project) |
| 1093 | for (const StringRef A : ProjectLevelArgs) |
| 1094 | ArgStrings.emplace_back(args: A); |
| 1095 | } |
| 1096 | |
| 1097 | } // namespace installapi |
| 1098 | } // namespace clang |
| 1099 | |