| /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying |
| file Copyright.txt or https://cmake.org/licensing for details. */ |
| #include "cmAddExecutableCommand.h" |
| |
| #include "cmExecutionStatus.h" |
| #include "cmGeneratorExpression.h" |
| #include "cmGlobalGenerator.h" |
| #include "cmMakefile.h" |
| #include "cmStateTypes.h" |
| #include "cmStringAlgorithms.h" |
| #include "cmTarget.h" |
| |
| bool cmAddExecutableCommand(std::vector<std::string> const& args, |
| cmExecutionStatus& status) |
| { |
| if (args.empty()) { |
| status.SetError("called with incorrect number of arguments"); |
| return false; |
| } |
| |
| cmMakefile& mf = status.GetMakefile(); |
| auto s = args.begin(); |
| |
| std::string const& exename = *s; |
| |
| ++s; |
| bool use_win32 = false; |
| bool use_macbundle = false; |
| bool excludeFromAll = false; |
| bool importTarget = false; |
| bool importGlobal = false; |
| bool isAlias = false; |
| while (s != args.end()) { |
| if (*s == "WIN32") { |
| ++s; |
| use_win32 = true; |
| } else if (*s == "MACOSX_BUNDLE") { |
| ++s; |
| use_macbundle = true; |
| } else if (*s == "EXCLUDE_FROM_ALL") { |
| ++s; |
| excludeFromAll = true; |
| } else if (*s == "IMPORTED") { |
| ++s; |
| importTarget = true; |
| } else if (importTarget && *s == "GLOBAL") { |
| ++s; |
| importGlobal = true; |
| } else if (*s == "ALIAS") { |
| ++s; |
| isAlias = true; |
| } else { |
| break; |
| } |
| } |
| |
| bool nameOk = cmGeneratorExpression::IsValidTargetName(exename) && |
| !cmGlobalGenerator::IsReservedTarget(exename); |
| |
| if (nameOk && !importTarget && !isAlias) { |
| nameOk = exename.find(':') == std::string::npos; |
| } |
| if (!nameOk && !mf.CheckCMP0037(exename, cmStateEnums::EXECUTABLE)) { |
| return false; |
| } |
| |
| // Special modifiers are not allowed with IMPORTED signature. |
| if (importTarget && (use_win32 || use_macbundle || excludeFromAll)) { |
| if (use_win32) { |
| status.SetError("may not be given WIN32 for an IMPORTED target."); |
| } else if (use_macbundle) { |
| status.SetError( |
| "may not be given MACOSX_BUNDLE for an IMPORTED target."); |
| } else // if(excludeFromAll) |
| { |
| status.SetError( |
| "may not be given EXCLUDE_FROM_ALL for an IMPORTED target."); |
| } |
| return false; |
| } |
| if (isAlias) { |
| if (!cmGeneratorExpression::IsValidTargetName(exename)) { |
| status.SetError("Invalid name for ALIAS: " + exename); |
| return false; |
| } |
| if (excludeFromAll) { |
| status.SetError("EXCLUDE_FROM_ALL with ALIAS makes no sense."); |
| return false; |
| } |
| if (importTarget || importGlobal) { |
| status.SetError("IMPORTED with ALIAS is not allowed."); |
| return false; |
| } |
| if (args.size() != 3) { |
| status.SetError("ALIAS requires exactly one target argument."); |
| return false; |
| } |
| |
| std::string const& aliasedName = *s; |
| if (mf.IsAlias(aliasedName)) { |
| status.SetError(cmStrCat("cannot create ALIAS target \"", exename, |
| "\" because target \"", aliasedName, |
| "\" is itself an ALIAS.")); |
| return false; |
| } |
| cmTarget* aliasedTarget = mf.FindTargetToUse(aliasedName, true); |
| if (!aliasedTarget) { |
| status.SetError(cmStrCat("cannot create ALIAS target \"", exename, |
| "\" because target \"", aliasedName, |
| "\" does not already exist.")); |
| return false; |
| } |
| cmStateEnums::TargetType type = aliasedTarget->GetType(); |
| if (type != cmStateEnums::EXECUTABLE) { |
| status.SetError(cmStrCat("cannot create ALIAS target \"", exename, |
| "\" because target \"", aliasedName, |
| "\" is not an executable.")); |
| return false; |
| } |
| mf.AddAlias(exename, aliasedName, |
| !aliasedTarget->IsImported() || |
| aliasedTarget->IsImportedGloballyVisible()); |
| return true; |
| } |
| |
| // Handle imported target creation. |
| if (importTarget) { |
| // Make sure the target does not already exist. |
| if (mf.FindTargetToUse(exename)) { |
| status.SetError(cmStrCat( |
| "cannot create imported target \"", exename, |
| "\" because another target with the same name already exists.")); |
| return false; |
| } |
| |
| // Create the imported target. |
| mf.AddImportedTarget(exename, cmStateEnums::EXECUTABLE, importGlobal); |
| return true; |
| } |
| |
| // Enforce name uniqueness. |
| { |
| std::string msg; |
| if (!mf.EnforceUniqueName(exename, msg)) { |
| status.SetError(msg); |
| return false; |
| } |
| } |
| |
| std::vector<std::string> srclists(s, args.end()); |
| cmTarget* tgt = mf.AddExecutable(exename, srclists, excludeFromAll); |
| if (use_win32) { |
| tgt->SetProperty("WIN32_EXECUTABLE", "ON"); |
| } |
| if (use_macbundle) { |
| tgt->SetProperty("MACOSX_BUNDLE", "ON"); |
| } |
| |
| return true; |
| } |