| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523 |
- /*============================================================================
- CMake - Cross Platform Makefile Generator
- Copyright 2000-2012 Kitware, Inc., Insight Software Consortium
- Distributed under the OSI-approved BSD License (the "License");
- see accompanying file Copyright.txt for details.
- This software is distributed WITHOUT ANY WARRANTY; without even the
- implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- See the License for more information.
- ============================================================================*/
- #include "cmGeneratorTarget.h"
- #include "cmTarget.h"
- #include "cmMakefile.h"
- #include "cmLocalGenerator.h"
- #include "cmGlobalGenerator.h"
- #include "cmSourceFile.h"
- #include "cmGeneratorExpression.h"
- #include "cmGeneratorExpressionDAGChecker.h"
- #include "cmComputeLinkInformation.h"
- #include "cmCustomCommandGenerator.h"
- #include "cmAlgorithms.h"
- #include <queue>
- #include <errno.h>
- #include "assert.h"
- //----------------------------------------------------------------------------
- void reportBadObjLib(std::vector<cmSourceFile*> const& badObjLib,
- cmTarget *target, cmake *cm)
- {
- if(!badObjLib.empty())
- {
- std::ostringstream e;
- e << "OBJECT library \"" << target->GetName() << "\" contains:\n";
- for(std::vector<cmSourceFile*>::const_iterator i = badObjLib.begin();
- i != badObjLib.end(); ++i)
- {
- e << " " << (*i)->GetLocation().GetName() << "\n";
- }
- e << "but may contain only sources that compile, header files, and "
- "other files that would not affect linking of a normal library.";
- cm->IssueMessage(cmake::FATAL_ERROR, e.str(),
- target->GetBacktrace());
- }
- }
- struct ObjectSourcesTag {};
- struct CustomCommandsTag {};
- struct ExtraSourcesTag {};
- struct HeaderSourcesTag {};
- struct ExternalObjectsTag {};
- struct IDLSourcesTag {};
- struct ResxTag {};
- struct ModuleDefinitionFileTag {};
- struct AppManifestTag{};
- struct CertificatesTag{};
- struct XamlTag{};
- template<typename Tag, typename OtherTag>
- struct IsSameTag
- {
- enum {
- Result = false
- };
- };
- template<typename Tag>
- struct IsSameTag<Tag, Tag>
- {
- enum {
- Result = true
- };
- };
- template<bool>
- struct DoAccept
- {
- template <typename T> static void Do(T&, cmSourceFile*) {}
- };
- template<>
- struct DoAccept<true>
- {
- static void Do(std::vector<cmSourceFile const*>& files, cmSourceFile* f)
- {
- files.push_back(f);
- }
- static void Do(cmGeneratorTarget::ResxData& data, cmSourceFile* f)
- {
- // Build and save the name of the corresponding .h file
- // This relationship will be used later when building the project files.
- // Both names would have been auto generated from Visual Studio
- // where the user supplied the file name and Visual Studio
- // appended the suffix.
- std::string resx = f->GetFullPath();
- std::string hFileName = resx.substr(0, resx.find_last_of(".")) + ".h";
- data.ExpectedResxHeaders.insert(hFileName);
- data.ResxSources.push_back(f);
- }
- static void Do(cmGeneratorTarget::XamlData& data, cmSourceFile* f)
- {
- // Build and save the name of the corresponding .h and .cpp file
- // This relationship will be used later when building the project files.
- // Both names would have been auto generated from Visual Studio
- // where the user supplied the file name and Visual Studio
- // appended the suffix.
- std::string xaml = f->GetFullPath();
- std::string hFileName = xaml + ".h";
- std::string cppFileName = xaml + ".cpp";
- data.ExpectedXamlHeaders.insert(hFileName);
- data.ExpectedXamlSources.insert(cppFileName);
- data.XamlSources.push_back(f);
- }
- static void Do(std::string& data, cmSourceFile* f)
- {
- data = f->GetFullPath();
- }
- };
- //----------------------------------------------------------------------------
- template<typename Tag, typename DataType = std::vector<cmSourceFile const*> >
- struct TagVisitor
- {
- DataType& Data;
- std::vector<cmSourceFile*> BadObjLibFiles;
- cmTarget *Target;
- cmGlobalGenerator *GlobalGenerator;
- cmsys::RegularExpression Header;
- bool IsObjLib;
- TagVisitor(cmTarget *target, DataType& data)
- : Data(data), Target(target),
- GlobalGenerator(target->GetMakefile()->GetGlobalGenerator()),
- Header(CM_HEADER_REGEX),
- IsObjLib(target->GetType() == cmTarget::OBJECT_LIBRARY)
- {
- }
- ~TagVisitor()
- {
- reportBadObjLib(this->BadObjLibFiles, this->Target,
- this->GlobalGenerator->GetCMakeInstance());
- }
- void Accept(cmSourceFile *sf)
- {
- std::string ext = cmSystemTools::LowerCase(sf->GetExtension());
- if(sf->GetCustomCommand())
- {
- DoAccept<IsSameTag<Tag, CustomCommandsTag>::Result>::Do(this->Data, sf);
- }
- else if(this->Target->GetType() == cmTarget::UTILITY)
- {
- DoAccept<IsSameTag<Tag, ExtraSourcesTag>::Result>::Do(this->Data, sf);
- }
- else if(sf->GetPropertyAsBool("HEADER_FILE_ONLY"))
- {
- DoAccept<IsSameTag<Tag, HeaderSourcesTag>::Result>::Do(this->Data, sf);
- }
- else if(sf->GetPropertyAsBool("EXTERNAL_OBJECT"))
- {
- DoAccept<IsSameTag<Tag, ExternalObjectsTag>::Result>::Do(this->Data, sf);
- if(this->IsObjLib)
- {
- this->BadObjLibFiles.push_back(sf);
- }
- }
- else if(!sf->GetLanguage().empty())
- {
- DoAccept<IsSameTag<Tag, ObjectSourcesTag>::Result>::Do(this->Data, sf);
- }
- else if(ext == "def")
- {
- DoAccept<IsSameTag<Tag, ModuleDefinitionFileTag>::Result>::Do(this->Data,
- sf);
- if(this->IsObjLib)
- {
- this->BadObjLibFiles.push_back(sf);
- }
- }
- else if(ext == "idl")
- {
- DoAccept<IsSameTag<Tag, IDLSourcesTag>::Result>::Do(this->Data, sf);
- if(this->IsObjLib)
- {
- this->BadObjLibFiles.push_back(sf);
- }
- }
- else if(ext == "resx")
- {
- DoAccept<IsSameTag<Tag, ResxTag>::Result>::Do(this->Data, sf);
- }
- else if (ext == "appxmanifest")
- {
- DoAccept<IsSameTag<Tag, AppManifestTag>::Result>::Do(this->Data, sf);
- }
- else if (ext == "pfx")
- {
- DoAccept<IsSameTag<Tag, CertificatesTag>::Result>::Do(this->Data, sf);
- }
- else if (ext == "xaml")
- {
- DoAccept<IsSameTag<Tag, XamlTag>::Result>::Do(this->Data, sf);
- }
- else if(this->Header.find(sf->GetFullPath().c_str()))
- {
- DoAccept<IsSameTag<Tag, HeaderSourcesTag>::Result>::Do(this->Data, sf);
- }
- else if(this->GlobalGenerator->IgnoreFile(sf->GetExtension().c_str()))
- {
- DoAccept<IsSameTag<Tag, ExtraSourcesTag>::Result>::Do(this->Data, sf);
- }
- else
- {
- DoAccept<IsSameTag<Tag, ExtraSourcesTag>::Result>::Do(this->Data, sf);
- }
- }
- };
- //----------------------------------------------------------------------------
- cmGeneratorTarget::cmGeneratorTarget(cmTarget* t, cmLocalGenerator* lg)
- : Target(t),
- SourceFileFlagsConstructed(false)
- {
- this->Makefile = this->Target->GetMakefile();
- this->LocalGenerator = lg;
- this->GlobalGenerator = this->Makefile->GetGlobalGenerator();
- }
- cmGeneratorTarget::~cmGeneratorTarget()
- {
- cmDeleteAll(this->LinkInformation);
- this->LinkInformation.clear();
- }
- cmLocalGenerator* cmGeneratorTarget::GetLocalGenerator() const
- {
- return this->LocalGenerator;
- }
- //----------------------------------------------------------------------------
- int cmGeneratorTarget::GetType() const
- {
- return this->Target->GetType();
- }
- //----------------------------------------------------------------------------
- std::string cmGeneratorTarget::GetName() const
- {
- return this->Target->GetName();
- }
- //----------------------------------------------------------------------------
- const char *cmGeneratorTarget::GetProperty(const std::string& prop) const
- {
- return this->Target->GetProperty(prop);
- }
- //----------------------------------------------------------------------------
- std::vector<cmSourceFile*> const*
- cmGeneratorTarget::GetSourceDepends(cmSourceFile const* sf) const
- {
- SourceEntriesType::const_iterator i = this->SourceEntries.find(sf);
- if(i != this->SourceEntries.end())
- {
- return &i->second.Depends;
- }
- return 0;
- }
- static void handleSystemIncludesDep(cmMakefile *mf, cmTarget const* depTgt,
- const std::string& config,
- cmTarget *headTarget,
- cmGeneratorExpressionDAGChecker *dagChecker,
- std::vector<std::string>& result,
- bool excludeImported)
- {
- if (const char* dirs =
- depTgt->GetProperty("INTERFACE_SYSTEM_INCLUDE_DIRECTORIES"))
- {
- cmGeneratorExpression ge;
- cmSystemTools::ExpandListArgument(ge.Parse(dirs)
- ->Evaluate(mf,
- config, false, headTarget,
- depTgt, dagChecker), result);
- }
- if (!depTgt->IsImported() || excludeImported)
- {
- return;
- }
- if (const char* dirs =
- depTgt->GetProperty("INTERFACE_INCLUDE_DIRECTORIES"))
- {
- cmGeneratorExpression ge;
- cmSystemTools::ExpandListArgument(ge.Parse(dirs)
- ->Evaluate(mf,
- config, false, headTarget,
- depTgt, dagChecker), result);
- }
- }
- #define IMPLEMENT_VISIT_IMPL(DATA, DATATYPE) \
- { \
- std::vector<cmSourceFile*> sourceFiles; \
- this->Target->GetSourceFiles(sourceFiles, config); \
- TagVisitor<DATA ## Tag DATATYPE> visitor(this->Target, data); \
- for(std::vector<cmSourceFile*>::const_iterator si = sourceFiles.begin(); \
- si != sourceFiles.end(); ++si) \
- { \
- visitor.Accept(*si); \
- } \
- } \
- #define IMPLEMENT_VISIT(DATA) \
- IMPLEMENT_VISIT_IMPL(DATA, EMPTY) \
- #define EMPTY
- #define COMMA ,
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetObjectSources(std::vector<cmSourceFile const*> &data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(ObjectSources);
- if (!this->Objects.empty())
- {
- return;
- }
- for(std::vector<cmSourceFile const*>::const_iterator it = data.begin();
- it != data.end(); ++it)
- {
- this->Objects[*it];
- }
- this->LocalGenerator->ComputeObjectFilenames(this->Objects, this);
- }
- void cmGeneratorTarget::ComputeObjectMapping()
- {
- if(!this->Objects.empty())
- {
- return;
- }
- std::vector<std::string> configs;
- this->Makefile->GetConfigurations(configs);
- if (configs.empty())
- {
- configs.push_back("");
- }
- for(std::vector<std::string>::const_iterator ci = configs.begin();
- ci != configs.end(); ++ci)
- {
- std::vector<cmSourceFile const*> sourceFiles;
- this->GetObjectSources(sourceFiles, *ci);
- }
- }
- //----------------------------------------------------------------------------
- const char* cmGeneratorTarget::GetFeature(const std::string& feature,
- const std::string& config) const
- {
- if(!config.empty())
- {
- std::string featureConfig = feature;
- featureConfig += "_";
- featureConfig += cmSystemTools::UpperCase(config);
- if(const char* value = this->Target->GetProperty(featureConfig))
- {
- return value;
- }
- }
- if(const char* value = this->Target->GetProperty(feature))
- {
- return value;
- }
- return this->LocalGenerator->GetFeature(feature, config);
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::GetFeatureAsBool(const std::string& feature,
- const std::string& config) const
- {
- return cmSystemTools::IsOn(this->GetFeature(feature, config));
- }
- //----------------------------------------------------------------------------
- const std::string& cmGeneratorTarget::GetObjectName(cmSourceFile const* file)
- {
- this->ComputeObjectMapping();
- return this->Objects[file];
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::AddExplicitObjectName(cmSourceFile const* sf)
- {
- this->ExplicitObjectName.insert(sf);
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::HasExplicitObjectName(cmSourceFile const* file) const
- {
- const_cast<cmGeneratorTarget*>(this)->ComputeObjectMapping();
- std::set<cmSourceFile const*>::const_iterator it
- = this->ExplicitObjectName.find(file);
- return it != this->ExplicitObjectName.end();
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget
- ::GetIDLSources(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(IDLSources);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetHeaderSources(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(HeaderSources);
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget
- ::GetExtraSources(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(ExtraSources);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetCustomCommands(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(CustomCommands);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetExternalObjects(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(ExternalObjects);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget::GetExpectedResxHeaders(std::set<std::string>& srcs,
- const std::string& config) const
- {
- ResxData data;
- IMPLEMENT_VISIT_IMPL(Resx, COMMA cmGeneratorTarget::ResxData)
- srcs = data.ExpectedResxHeaders;
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget
- ::GetResxSources(std::vector<cmSourceFile const*>& srcs,
- const std::string& config) const
- {
- ResxData data;
- IMPLEMENT_VISIT_IMPL(Resx, COMMA cmGeneratorTarget::ResxData)
- srcs = data.ResxSources;
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetAppManifest(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(AppManifest);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget
- ::GetCertificates(std::vector<cmSourceFile const*>& data,
- const std::string& config) const
- {
- IMPLEMENT_VISIT(Certificates);
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget::GetExpectedXamlHeaders(std::set<std::string>& headers,
- const std::string& config) const
- {
- XamlData data;
- IMPLEMENT_VISIT_IMPL(Xaml, COMMA cmGeneratorTarget::XamlData)
- headers = data.ExpectedXamlHeaders;
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget::GetExpectedXamlSources(std::set<std::string>& srcs,
- const std::string& config) const
- {
- XamlData data;
- IMPLEMENT_VISIT_IMPL(Xaml, COMMA cmGeneratorTarget::XamlData)
- srcs = data.ExpectedXamlSources;
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget
- ::GetXamlSources(std::vector<cmSourceFile const*>& srcs,
- const std::string& config) const
- {
- XamlData data;
- IMPLEMENT_VISIT_IMPL(Xaml, COMMA cmGeneratorTarget::XamlData)
- srcs = data.XamlSources;
- }
- //----------------------------------------------------------------------------
- const char* cmGeneratorTarget::GetLocation(const std::string& config) const
- {
- static std::string location;
- if (this->Target->IsImported())
- {
- location = this->Target->ImportedGetFullPath(config, false);
- }
- else
- {
- location = this->GetFullPath(config, false);
- }
- return location.c_str();
- }
- bool cmGeneratorTarget::IsImported() const
- {
- return this->Target->IsImported();
- }
- //----------------------------------------------------------------------------
- const char* cmGeneratorTarget::GetLocationForBuild() const
- {
- static std::string location;
- if(this->IsImported())
- {
- location = this->Target->ImportedGetFullPath("", false);
- return location.c_str();
- }
- // Now handle the deprecated build-time configuration location.
- location = this->Target->GetDirectory();
- const char* cfgid = this->Makefile->GetDefinition("CMAKE_CFG_INTDIR");
- if(cfgid && strcmp(cfgid, ".") != 0)
- {
- location += "/";
- location += cfgid;
- }
- if(this->Target->IsAppBundleOnApple())
- {
- std::string macdir = this->Target->BuildMacContentDirectory("", "",
- false);
- if(!macdir.empty())
- {
- location += "/";
- location += macdir;
- }
- }
- location += "/";
- location += this->Target->GetFullName("", false);
- return location.c_str();
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::IsSystemIncludeDirectory(const std::string& dir,
- const std::string& config) const
- {
- assert(this->GetType() != cmTarget::INTERFACE_LIBRARY);
- std::string config_upper;
- if(!config.empty())
- {
- config_upper = cmSystemTools::UpperCase(config);
- }
- typedef std::map<std::string, std::vector<std::string> > IncludeCacheType;
- IncludeCacheType::const_iterator iter =
- this->SystemIncludesCache.find(config_upper);
- if (iter == this->SystemIncludesCache.end())
- {
- cmGeneratorExpressionDAGChecker dagChecker(
- this->GetName(),
- "SYSTEM_INCLUDE_DIRECTORIES", 0, 0);
- bool excludeImported
- = this->Target->GetPropertyAsBool("NO_SYSTEM_FROM_IMPORTED");
- std::vector<std::string> result;
- for (std::set<std::string>::const_iterator
- it = this->Target->GetSystemIncludeDirectories().begin();
- it != this->Target->GetSystemIncludeDirectories().end(); ++it)
- {
- cmGeneratorExpression ge;
- cmSystemTools::ExpandListArgument(ge.Parse(*it)
- ->Evaluate(this->Makefile,
- config, false, this->Target,
- &dagChecker), result);
- }
- std::vector<cmTarget const*> const& deps =
- this->GetLinkImplementationClosure(config);
- for(std::vector<cmTarget const*>::const_iterator
- li = deps.begin(), le = deps.end(); li != le; ++li)
- {
- handleSystemIncludesDep(this->Makefile, *li, config, this->Target,
- &dagChecker, result, excludeImported);
- }
- std::set<std::string> unique;
- for(std::vector<std::string>::iterator li = result.begin();
- li != result.end(); ++li)
- {
- cmSystemTools::ConvertToUnixSlashes(*li);
- unique.insert(*li);
- }
- result.clear();
- result.insert(result.end(), unique.begin(), unique.end());
- IncludeCacheType::value_type entry(config_upper, result);
- iter = this->SystemIncludesCache.insert(entry).first;
- }
- return std::binary_search(iter->second.begin(), iter->second.end(), dir);
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::GetPropertyAsBool(const std::string& prop) const
- {
- return this->Target->GetPropertyAsBool(prop);
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GetSourceFiles(std::vector<cmSourceFile*> &files,
- const std::string& config) const
- {
- this->Target->GetSourceFiles(files, config);
- }
- //----------------------------------------------------------------------------
- std::string
- cmGeneratorTarget::GetCompilePDBName(const std::string& config) const
- {
- std::string prefix;
- std::string base;
- std::string suffix;
- this->Target->GetFullNameInternal(config, false, prefix, base, suffix);
- // Check for a per-configuration output directory target property.
- std::string configUpper = cmSystemTools::UpperCase(config);
- std::string configProp = "COMPILE_PDB_NAME_";
- configProp += configUpper;
- const char* config_name = this->Target->GetProperty(configProp);
- if(config_name && *config_name)
- {
- return prefix + config_name + ".pdb";
- }
- const char* name = this->Target->GetProperty("COMPILE_PDB_NAME");
- if(name && *name)
- {
- return prefix + name + ".pdb";
- }
- return "";
- }
- //----------------------------------------------------------------------------
- std::string
- cmGeneratorTarget::GetCompilePDBPath(const std::string& config) const
- {
- std::string dir = this->Target->GetCompilePDBDirectory(config);
- std::string name = this->GetCompilePDBName(config);
- if(dir.empty() && !name.empty())
- {
- dir = this->Target->GetPDBDirectory(config);
- }
- if(!dir.empty())
- {
- dir += "/";
- }
- return dir + name;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::HasSOName(const std::string& config) const
- {
- // soname is supported only for shared libraries and modules,
- // and then only when the platform supports an soname flag.
- return ((this->GetType() == cmTarget::SHARED_LIBRARY ||
- this->GetType() == cmTarget::MODULE_LIBRARY) &&
- !this->GetPropertyAsBool("NO_SONAME") &&
- this->Makefile->GetSONameFlag(
- this->Target->GetLinkerLanguage(config)));
- }
- //----------------------------------------------------------------------------
- std::string cmGeneratorTarget::GetSOName(const std::string& config) const
- {
- if(this->Target->IsImported())
- {
- // Lookup the imported soname.
- if(cmTarget::ImportInfo const* info = this->Target->GetImportInfo(config))
- {
- if(info->NoSOName)
- {
- // The imported library has no builtin soname so the name
- // searched at runtime will be just the filename.
- return cmSystemTools::GetFilenameName(info->Location);
- }
- else
- {
- // Use the soname given if any.
- if(info->SOName.find("@rpath/") == 0)
- {
- return info->SOName.substr(6);
- }
- return info->SOName;
- }
- }
- else
- {
- return "";
- }
- }
- else
- {
- // Compute the soname that will be built.
- std::string name;
- std::string soName;
- std::string realName;
- std::string impName;
- std::string pdbName;
- this->GetLibraryNames(name, soName, realName,
- impName, pdbName, config);
- return soName;
- }
- }
- //----------------------------------------------------------------------------
- std::string
- cmGeneratorTarget::GetModuleDefinitionFile(const std::string& config) const
- {
- std::string data;
- IMPLEMENT_VISIT_IMPL(ModuleDefinitionFile, COMMA std::string)
- return data;
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget::UseObjectLibraries(std::vector<std::string>& objs,
- const std::string &config) const
- {
- std::vector<cmSourceFile const*> objectFiles;
- this->GetExternalObjects(objectFiles, config);
- std::vector<cmTarget*> objectLibraries;
- for(std::vector<cmSourceFile const*>::const_iterator
- it = objectFiles.begin(); it != objectFiles.end(); ++it)
- {
- std::string objLib = (*it)->GetObjectLibrary();
- if (cmTarget* tgt = this->Makefile->FindTargetToUse(objLib))
- {
- objectLibraries.push_back(tgt);
- }
- }
- std::vector<cmTarget*>::const_iterator end
- = cmRemoveDuplicates(objectLibraries);
- for(std::vector<cmTarget*>::const_iterator
- ti = objectLibraries.begin();
- ti != end; ++ti)
- {
- cmTarget* objLib = *ti;
- cmGeneratorTarget* ogt =
- this->GlobalGenerator->GetGeneratorTarget(objLib);
- std::vector<cmSourceFile const*> objectSources;
- ogt->GetObjectSources(objectSources, config);
- for(std::vector<cmSourceFile const*>::const_iterator
- si = objectSources.begin();
- si != objectSources.end(); ++si)
- {
- std::string obj = ogt->ObjectDirectory;
- obj += ogt->Objects[*si];
- objs.push_back(obj);
- }
- }
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GetAutoUicOptions(std::vector<std::string> &result,
- const std::string& config) const
- {
- const char *prop
- = this->GetLinkInterfaceDependentStringProperty("AUTOUIC_OPTIONS",
- config);
- if (!prop)
- {
- return;
- }
- cmGeneratorExpression ge;
- cmGeneratorExpressionDAGChecker dagChecker(
- this->GetName(),
- "AUTOUIC_OPTIONS", 0, 0);
- cmSystemTools::ExpandListArgument(ge.Parse(prop)
- ->Evaluate(this->Makefile,
- config,
- false,
- this->Target,
- &dagChecker),
- result);
- }
- //----------------------------------------------------------------------------
- void processILibs(const std::string& config,
- cmTarget const* headTarget,
- cmLinkItem const& item,
- std::vector<cmTarget const*>& tgts,
- std::set<cmTarget const*>& emitted)
- {
- if (item.Target && emitted.insert(item.Target).second)
- {
- tgts.push_back(item.Target);
- if(cmTarget::LinkInterfaceLibraries const* iface =
- item.Target->GetLinkInterfaceLibraries(config, headTarget, true))
- {
- for(std::vector<cmLinkItem>::const_iterator
- it = iface->Libraries.begin();
- it != iface->Libraries.end(); ++it)
- {
- processILibs(config, headTarget, *it, tgts, emitted);
- }
- }
- }
- }
- //----------------------------------------------------------------------------
- const std::vector<const cmTarget*>&
- cmGeneratorTarget::GetLinkImplementationClosure(
- const std::string& config) const
- {
- LinkImplClosure& tgts =
- this->LinkImplClosureMap[config];
- if(!tgts.Done)
- {
- tgts.Done = true;
- std::set<cmTarget const*> emitted;
- cmTarget::LinkImplementationLibraries const* impl
- = this->Target->GetLinkImplementationLibraries(config);
- for(std::vector<cmLinkImplItem>::const_iterator
- it = impl->Libraries.begin();
- it != impl->Libraries.end(); ++it)
- {
- processILibs(config, this->Target, *it, tgts , emitted);
- }
- }
- return tgts;
- }
- //----------------------------------------------------------------------------
- class cmTargetTraceDependencies
- {
- public:
- cmTargetTraceDependencies(cmGeneratorTarget* target);
- void Trace();
- private:
- cmTarget* Target;
- cmGeneratorTarget* GeneratorTarget;
- cmMakefile* Makefile;
- cmGlobalGenerator const* GlobalGenerator;
- typedef cmGeneratorTarget::SourceEntry SourceEntry;
- SourceEntry* CurrentEntry;
- std::queue<cmSourceFile*> SourceQueue;
- std::set<cmSourceFile*> SourcesQueued;
- typedef std::map<std::string, cmSourceFile*> NameMapType;
- NameMapType NameMap;
- std::vector<std::string> NewSources;
- void QueueSource(cmSourceFile* sf);
- void FollowName(std::string const& name);
- void FollowNames(std::vector<std::string> const& names);
- bool IsUtility(std::string const& dep);
- void CheckCustomCommand(cmCustomCommand const& cc);
- void CheckCustomCommands(const std::vector<cmCustomCommand>& commands);
- void FollowCommandDepends(cmCustomCommand const& cc,
- const std::string& config,
- std::set<std::string>& emitted);
- };
- //----------------------------------------------------------------------------
- cmTargetTraceDependencies
- ::cmTargetTraceDependencies(cmGeneratorTarget* target):
- Target(target->Target), GeneratorTarget(target)
- {
- // Convenience.
- this->Makefile = this->Target->GetMakefile();
- this->GlobalGenerator = this->Makefile->GetGlobalGenerator();
- this->CurrentEntry = 0;
- // Queue all the source files already specified for the target.
- if (this->Target->GetType() != cmTarget::INTERFACE_LIBRARY)
- {
- std::vector<std::string> configs;
- this->Makefile->GetConfigurations(configs);
- if (configs.empty())
- {
- configs.push_back("");
- }
- std::set<cmSourceFile*> emitted;
- for(std::vector<std::string>::const_iterator ci = configs.begin();
- ci != configs.end(); ++ci)
- {
- std::vector<cmSourceFile*> sources;
- this->Target->GetSourceFiles(sources, *ci);
- for(std::vector<cmSourceFile*>::const_iterator si = sources.begin();
- si != sources.end(); ++si)
- {
- cmSourceFile* sf = *si;
- const std::set<cmTarget const*> tgts =
- this->GlobalGenerator->GetFilenameTargetDepends(sf);
- if (tgts.find(this->Target) != tgts.end())
- {
- std::ostringstream e;
- e << "Evaluation output file\n \"" << sf->GetFullPath()
- << "\"\ndepends on the sources of a target it is used in. This "
- "is a dependency loop and is not allowed.";
- this->GeneratorTarget
- ->LocalGenerator->IssueMessage(cmake::FATAL_ERROR, e.str());
- return;
- }
- if(emitted.insert(sf).second && this->SourcesQueued.insert(sf).second)
- {
- this->SourceQueue.push(sf);
- }
- }
- }
- }
- // Queue pre-build, pre-link, and post-build rule dependencies.
- this->CheckCustomCommands(this->Target->GetPreBuildCommands());
- this->CheckCustomCommands(this->Target->GetPreLinkCommands());
- this->CheckCustomCommands(this->Target->GetPostBuildCommands());
- }
- //----------------------------------------------------------------------------
- void cmTargetTraceDependencies::Trace()
- {
- // Process one dependency at a time until the queue is empty.
- while(!this->SourceQueue.empty())
- {
- // Get the next source from the queue.
- cmSourceFile* sf = this->SourceQueue.front();
- this->SourceQueue.pop();
- this->CurrentEntry = &this->GeneratorTarget->SourceEntries[sf];
- // Queue dependencies added explicitly by the user.
- if(const char* additionalDeps = sf->GetProperty("OBJECT_DEPENDS"))
- {
- std::vector<std::string> objDeps;
- cmSystemTools::ExpandListArgument(additionalDeps, objDeps);
- for(std::vector<std::string>::iterator odi = objDeps.begin();
- odi != objDeps.end(); ++odi)
- {
- if (cmSystemTools::FileIsFullPath(*odi))
- {
- *odi = cmSystemTools::CollapseFullPath(*odi);
- }
- }
- this->FollowNames(objDeps);
- }
- // Queue the source needed to generate this file, if any.
- this->FollowName(sf->GetFullPath());
- // Queue dependencies added programatically by commands.
- this->FollowNames(sf->GetDepends());
- // Queue custom command dependencies.
- if(cmCustomCommand const* cc = sf->GetCustomCommand())
- {
- this->CheckCustomCommand(*cc);
- }
- }
- this->CurrentEntry = 0;
- this->Target->AddTracedSources(this->NewSources);
- }
- //----------------------------------------------------------------------------
- void cmTargetTraceDependencies::QueueSource(cmSourceFile* sf)
- {
- if(this->SourcesQueued.insert(sf).second)
- {
- this->SourceQueue.push(sf);
- // Make sure this file is in the target at the end.
- this->NewSources.push_back(sf->GetFullPath());
- }
- }
- //----------------------------------------------------------------------------
- void cmTargetTraceDependencies::FollowName(std::string const& name)
- {
- NameMapType::iterator i = this->NameMap.find(name);
- if(i == this->NameMap.end())
- {
- // Check if we know how to generate this file.
- cmSourceFile* sf = this->Makefile->GetSourceFileWithOutput(name);
- NameMapType::value_type entry(name, sf);
- i = this->NameMap.insert(entry).first;
- }
- if(cmSourceFile* sf = i->second)
- {
- // Record the dependency we just followed.
- if(this->CurrentEntry)
- {
- this->CurrentEntry->Depends.push_back(sf);
- }
- this->QueueSource(sf);
- }
- }
- //----------------------------------------------------------------------------
- void
- cmTargetTraceDependencies::FollowNames(std::vector<std::string> const& names)
- {
- for(std::vector<std::string>::const_iterator i = names.begin();
- i != names.end(); ++i)
- {
- this->FollowName(*i);
- }
- }
- //----------------------------------------------------------------------------
- bool cmTargetTraceDependencies::IsUtility(std::string const& dep)
- {
- // Dependencies on targets (utilities) are supposed to be named by
- // just the target name. However for compatibility we support
- // naming the output file generated by the target (assuming there is
- // no output-name property which old code would not have set). In
- // that case the target name will be the file basename of the
- // dependency.
- std::string util = cmSystemTools::GetFilenameName(dep);
- if(cmSystemTools::GetFilenameLastExtension(util) == ".exe")
- {
- util = cmSystemTools::GetFilenameWithoutLastExtension(util);
- }
- // Check for a target with this name.
- if(cmGeneratorTarget* t
- = this->Makefile->FindGeneratorTargetToUse(util))
- {
- // If we find the target and the dep was given as a full path,
- // then make sure it was not a full path to something else, and
- // the fact that the name matched a target was just a coincidence.
- if(cmSystemTools::FileIsFullPath(dep.c_str()))
- {
- if(t->GetType() >= cmTarget::EXECUTABLE &&
- t->GetType() <= cmTarget::MODULE_LIBRARY)
- {
- // This is really only for compatibility so we do not need to
- // worry about configuration names and output names.
- std::string tLocation = t->GetLocationForBuild();
- tLocation = cmSystemTools::GetFilenamePath(tLocation);
- std::string depLocation = cmSystemTools::GetFilenamePath(dep);
- depLocation = cmSystemTools::CollapseFullPath(depLocation);
- tLocation = cmSystemTools::CollapseFullPath(tLocation);
- if(depLocation == tLocation)
- {
- this->Target->AddUtility(util);
- return true;
- }
- }
- }
- else
- {
- // The original name of the dependency was not a full path. It
- // must name a target, so add the target-level dependency.
- this->Target->AddUtility(util);
- return true;
- }
- }
- // The dependency does not name a target built in this project.
- return false;
- }
- //----------------------------------------------------------------------------
- void
- cmTargetTraceDependencies
- ::CheckCustomCommand(cmCustomCommand const& cc)
- {
- // Transform command names that reference targets built in this
- // project to corresponding target-level dependencies.
- cmGeneratorExpression ge(cc.GetBacktrace());
- // Add target-level dependencies referenced by generator expressions.
- std::set<cmTarget*> targets;
- for(cmCustomCommandLines::const_iterator cit = cc.GetCommandLines().begin();
- cit != cc.GetCommandLines().end(); ++cit)
- {
- std::string const& command = *cit->begin();
- // Check for a target with this name.
- if(cmTarget* t = this->Makefile->FindTargetToUse(command))
- {
- if(t->GetType() == cmTarget::EXECUTABLE)
- {
- // The command refers to an executable target built in
- // this project. Add the target-level dependency to make
- // sure the executable is up to date before this custom
- // command possibly runs.
- this->Target->AddUtility(command);
- }
- }
- // Check for target references in generator expressions.
- for(cmCustomCommandLine::const_iterator cli = cit->begin();
- cli != cit->end(); ++cli)
- {
- const cmsys::auto_ptr<cmCompiledGeneratorExpression> cge
- = ge.Parse(*cli);
- cge->Evaluate(this->Makefile, "", true);
- std::set<cmTarget*> geTargets = cge->GetTargets();
- targets.insert(geTargets.begin(), geTargets.end());
- }
- }
- for(std::set<cmTarget*>::iterator ti = targets.begin();
- ti != targets.end(); ++ti)
- {
- this->Target->AddUtility((*ti)->GetName());
- }
- // Queue the custom command dependencies.
- std::vector<std::string> configs;
- std::set<std::string> emitted;
- this->Makefile->GetConfigurations(configs);
- if (configs.empty())
- {
- configs.push_back("");
- }
- for(std::vector<std::string>::const_iterator ci = configs.begin();
- ci != configs.end(); ++ci)
- {
- this->FollowCommandDepends(cc, *ci, emitted);
- }
- }
- //----------------------------------------------------------------------------
- void cmTargetTraceDependencies::FollowCommandDepends(cmCustomCommand const& cc,
- const std::string& config,
- std::set<std::string>& emitted)
- {
- cmCustomCommandGenerator ccg(cc, config,
- this->GeneratorTarget->LocalGenerator);
- const std::vector<std::string>& depends = ccg.GetDepends();
- for(std::vector<std::string>::const_iterator di = depends.begin();
- di != depends.end(); ++di)
- {
- std::string const& dep = *di;
- if(emitted.insert(dep).second)
- {
- if(!this->IsUtility(dep))
- {
- // The dependency does not name a target and may be a file we
- // know how to generate. Queue it.
- this->FollowName(dep);
- }
- }
- }
- }
- //----------------------------------------------------------------------------
- void
- cmTargetTraceDependencies
- ::CheckCustomCommands(const std::vector<cmCustomCommand>& commands)
- {
- for(std::vector<cmCustomCommand>::const_iterator cli = commands.begin();
- cli != commands.end(); ++cli)
- {
- this->CheckCustomCommand(*cli);
- }
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::TraceDependencies()
- {
- // CMake-generated targets have no dependencies to trace. Normally tracing
- // would find nothing anyway, but when building CMake itself the "install"
- // target command ends up referencing the "cmake" target but we do not
- // really want the dependency because "install" depend on "all" anyway.
- if(this->GetType() == cmTarget::GLOBAL_TARGET)
- {
- return;
- }
- // Use a helper object to trace the dependencies.
- cmTargetTraceDependencies tracer(this);
- tracer.Trace();
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GetAppleArchs(const std::string& config,
- std::vector<std::string>& archVec) const
- {
- const char* archs = 0;
- if(!config.empty())
- {
- std::string defVarName = "OSX_ARCHITECTURES_";
- defVarName += cmSystemTools::UpperCase(config);
- archs = this->Target->GetProperty(defVarName);
- }
- if(!archs)
- {
- archs = this->Target->GetProperty("OSX_ARCHITECTURES");
- }
- if(archs)
- {
- cmSystemTools::ExpandListArgument(std::string(archs), archVec);
- }
- }
- //----------------------------------------------------------------------------
- std::string
- cmGeneratorTarget::GetCreateRuleVariable(std::string const& lang,
- std::string const& config) const
- {
- switch(this->GetType())
- {
- case cmTarget::STATIC_LIBRARY:
- {
- std::string var = "CMAKE_" + lang + "_CREATE_STATIC_LIBRARY";
- if(this->GetFeatureAsBool(
- "INTERPROCEDURAL_OPTIMIZATION", config))
- {
- std::string varIPO = var + "_IPO";
- if(this->Makefile->GetDefinition(varIPO))
- {
- return varIPO;
- }
- }
- return var;
- }
- case cmTarget::SHARED_LIBRARY:
- return "CMAKE_" + lang + "_CREATE_SHARED_LIBRARY";
- case cmTarget::MODULE_LIBRARY:
- return "CMAKE_" + lang + "_CREATE_SHARED_MODULE";
- case cmTarget::EXECUTABLE:
- return "CMAKE_" + lang + "_LINK_EXECUTABLE";
- default:
- break;
- }
- return "";
- }
- //----------------------------------------------------------------------------
- std::vector<std::string>
- cmGeneratorTarget::GetIncludeDirectories(const std::string& config,
- const std::string& lang) const
- {
- return this->Target->GetIncludeDirectories(config, lang);
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GenerateTargetManifest(
- const std::string& config) const
- {
- if (this->Target->IsImported())
- {
- return;
- }
- cmMakefile* mf = this->Target->GetMakefile();
- cmGlobalGenerator* gg = mf->GetGlobalGenerator();
- // Get the names.
- std::string name;
- std::string soName;
- std::string realName;
- std::string impName;
- std::string pdbName;
- if(this->GetType() == cmTarget::EXECUTABLE)
- {
- this->GetExecutableNames(name, realName, impName, pdbName, config);
- }
- else if(this->GetType() == cmTarget::STATIC_LIBRARY ||
- this->GetType() == cmTarget::SHARED_LIBRARY ||
- this->GetType() == cmTarget::MODULE_LIBRARY)
- {
- this->GetLibraryNames(name, soName, realName, impName, pdbName,
- config);
- }
- else
- {
- return;
- }
- // Get the directory.
- std::string dir = this->Target->GetDirectory(config, false);
- // Add each name.
- std::string f;
- if(!name.empty())
- {
- f = dir;
- f += "/";
- f += name;
- gg->AddToManifest(f);
- }
- if(!soName.empty())
- {
- f = dir;
- f += "/";
- f += soName;
- gg->AddToManifest(f);
- }
- if(!realName.empty())
- {
- f = dir;
- f += "/";
- f += realName;
- gg->AddToManifest(f);
- }
- if(!pdbName.empty())
- {
- f = dir;
- f += "/";
- f += pdbName;
- gg->AddToManifest(f);
- }
- if(!impName.empty())
- {
- f = this->Target->GetDirectory(config, true);
- f += "/";
- f += impName;
- gg->AddToManifest(f);
- }
- }
- //----------------------------------------------------------------------------
- std::string cmGeneratorTarget::GetFullPath(const std::string& config,
- bool implib, bool realname) const
- {
- if(this->Target->IsImported())
- {
- return this->Target->ImportedGetFullPath(config, implib);
- }
- else
- {
- return this->NormalGetFullPath(config, implib, realname);
- }
- }
- std::string cmGeneratorTarget::NormalGetFullPath(const std::string& config,
- bool implib,
- bool realname) const
- {
- std::string fpath = this->Target->GetDirectory(config, implib);
- fpath += "/";
- if(this->Target->IsAppBundleOnApple())
- {
- fpath = this->Target->BuildMacContentDirectory(fpath, config, false);
- fpath += "/";
- }
- // Add the full name of the target.
- if(implib)
- {
- fpath += this->Target->GetFullName(config, true);
- }
- else if(realname)
- {
- fpath += this->NormalGetRealName(config);
- }
- else
- {
- fpath += this->Target->GetFullName(config, false);
- }
- return fpath;
- }
- //----------------------------------------------------------------------------
- std::string
- cmGeneratorTarget::NormalGetRealName(const std::string& config) const
- {
- // This should not be called for imported targets.
- // TODO: Split cmTarget into a class hierarchy to get compile-time
- // enforcement of the limited imported target API.
- if(this->Target->IsImported())
- {
- std::string msg = "NormalGetRealName called on imported target: ";
- msg += this->GetName();
- this->Makefile->IssueMessage(cmake::INTERNAL_ERROR, msg);
- }
- if(this->GetType() == cmTarget::EXECUTABLE)
- {
- // Compute the real name that will be built.
- std::string name;
- std::string realName;
- std::string impName;
- std::string pdbName;
- this->GetExecutableNames(name, realName, impName, pdbName, config);
- return realName;
- }
- else
- {
- // Compute the real name that will be built.
- std::string name;
- std::string soName;
- std::string realName;
- std::string impName;
- std::string pdbName;
- this->GetLibraryNames(name, soName, realName,
- impName, pdbName, config);
- return realName;
- }
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GetLibraryNames(std::string& name,
- std::string& soName,
- std::string& realName,
- std::string& impName,
- std::string& pdbName,
- const std::string& config) const
- {
- // This should not be called for imported targets.
- // TODO: Split cmTarget into a class hierarchy to get compile-time
- // enforcement of the limited imported target API.
- if(this->Target->IsImported())
- {
- std::string msg = "GetLibraryNames called on imported target: ";
- msg += this->GetName();
- this->LocalGenerator->IssueMessage(cmake::INTERNAL_ERROR,
- msg);
- return;
- }
- // Check for library version properties.
- const char* version = this->GetProperty("VERSION");
- const char* soversion = this->GetProperty("SOVERSION");
- if(!this->HasSOName(config) ||
- this->Target->IsFrameworkOnApple())
- {
- // Versioning is supported only for shared libraries and modules,
- // and then only when the platform supports an soname flag.
- version = 0;
- soversion = 0;
- }
- if(version && !soversion)
- {
- // The soversion must be set if the library version is set. Use
- // the library version as the soversion.
- soversion = version;
- }
- if(!version && soversion)
- {
- // Use the soversion as the library version.
- version = soversion;
- }
- // Get the components of the library name.
- std::string prefix;
- std::string base;
- std::string suffix;
- this->Target->GetFullNameInternal(config, false, prefix, base, suffix);
- // The library name.
- name = prefix+base+suffix;
- if(this->Target->IsFrameworkOnApple())
- {
- realName = prefix;
- realName += "Versions/";
- realName += this->Target->GetFrameworkVersion();
- realName += "/";
- realName += base;
- soName = realName;
- }
- else
- {
- // The library's soname.
- this->Target->ComputeVersionedName(soName, prefix, base, suffix,
- name, soversion);
- // The library's real name on disk.
- this->Target->ComputeVersionedName(realName, prefix, base, suffix,
- name, version);
- }
- // The import library name.
- if(this->GetType() == cmTarget::SHARED_LIBRARY ||
- this->GetType() == cmTarget::MODULE_LIBRARY)
- {
- impName = this->Target->GetFullNameInternal(config, true);
- }
- else
- {
- impName = "";
- }
- // The program database file name.
- pdbName = this->GetPDBName(config);
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::GetExecutableNames(std::string& name,
- std::string& realName,
- std::string& impName,
- std::string& pdbName,
- const std::string& config) const
- {
- // This should not be called for imported targets.
- // TODO: Split cmTarget into a class hierarchy to get compile-time
- // enforcement of the limited imported target API.
- if(this->Target->IsImported())
- {
- std::string msg =
- "GetExecutableNames called on imported target: ";
- msg += this->GetName();
- this->LocalGenerator->IssueMessage(cmake::INTERNAL_ERROR, msg);
- }
- // This versioning is supported only for executables and then only
- // when the platform supports symbolic links.
- #if defined(_WIN32) && !defined(__CYGWIN__)
- const char* version = 0;
- #else
- // Check for executable version properties.
- const char* version = this->GetProperty("VERSION");
- if(this->GetType() != cmTarget::EXECUTABLE || this->Makefile->IsOn("XCODE"))
- {
- version = 0;
- }
- #endif
- // Get the components of the executable name.
- std::string prefix;
- std::string base;
- std::string suffix;
- this->Target->GetFullNameInternal(config, false, prefix, base, suffix);
- // The executable name.
- name = prefix+base+suffix;
- // The executable's real name on disk.
- #if defined(__CYGWIN__)
- realName = prefix+base;
- #else
- realName = name;
- #endif
- if(version)
- {
- realName += "-";
- realName += version;
- }
- #if defined(__CYGWIN__)
- realName += suffix;
- #endif
- // The import library name.
- impName = this->Target->GetFullNameInternal(config, true);
- // The program database file name.
- pdbName = this->GetPDBName(config);
- }
- //----------------------------------------------------------------------------
- std::string cmGeneratorTarget::GetPDBName(const std::string& config) const
- {
- std::string prefix;
- std::string base;
- std::string suffix;
- this->Target->GetFullNameInternal(config, false, prefix, base, suffix);
- std::vector<std::string> props;
- std::string configUpper =
- cmSystemTools::UpperCase(config);
- if(!configUpper.empty())
- {
- // PDB_NAME_<CONFIG>
- props.push_back("PDB_NAME_" + configUpper);
- }
- // PDB_NAME
- props.push_back("PDB_NAME");
- for(std::vector<std::string>::const_iterator i = props.begin();
- i != props.end(); ++i)
- {
- if(const char* outName = this->GetProperty(*i))
- {
- base = outName;
- break;
- }
- }
- return prefix+base+".pdb";
- }
- bool cmStrictTargetComparison::operator()(cmTarget const* t1,
- cmTarget const* t2) const
- {
- int nameResult = strcmp(t1->GetName().c_str(), t2->GetName().c_str());
- if (nameResult == 0)
- {
- return strcmp(t1->GetMakefile()->GetCurrentBinaryDirectory(),
- t2->GetMakefile()->GetCurrentBinaryDirectory()) < 0;
- }
- return nameResult < 0;
- }
- //----------------------------------------------------------------------------
- struct cmGeneratorTarget::SourceFileFlags
- cmGeneratorTarget::GetTargetSourceFileFlags(const cmSourceFile* sf) const
- {
- struct SourceFileFlags flags;
- this->ConstructSourceFileFlags();
- std::map<cmSourceFile const*, SourceFileFlags>::iterator si =
- this->SourceFlagsMap.find(sf);
- if(si != this->SourceFlagsMap.end())
- {
- flags = si->second;
- }
- else
- {
- // Handle the MACOSX_PACKAGE_LOCATION property on source files that
- // were not listed in one of the other lists.
- if(const char* location = sf->GetProperty("MACOSX_PACKAGE_LOCATION"))
- {
- flags.MacFolder = location;
- if(strcmp(location, "Resources") == 0)
- {
- flags.Type = cmGeneratorTarget::SourceFileTypeResource;
- }
- else
- {
- flags.Type = cmGeneratorTarget::SourceFileTypeMacContent;
- }
- }
- }
- return flags;
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::ConstructSourceFileFlags() const
- {
- if(this->SourceFileFlagsConstructed)
- {
- return;
- }
- this->SourceFileFlagsConstructed = true;
- // Process public headers to mark the source files.
- if(const char* files = this->Target->GetProperty("PUBLIC_HEADER"))
- {
- std::vector<std::string> relFiles;
- cmSystemTools::ExpandListArgument(files, relFiles);
- for(std::vector<std::string>::iterator it = relFiles.begin();
- it != relFiles.end(); ++it)
- {
- if(cmSourceFile* sf = this->Makefile->GetSource(*it))
- {
- SourceFileFlags& flags = this->SourceFlagsMap[sf];
- flags.MacFolder = "Headers";
- flags.Type = cmGeneratorTarget::SourceFileTypePublicHeader;
- }
- }
- }
- // Process private headers after public headers so that they take
- // precedence if a file is listed in both.
- if(const char* files = this->Target->GetProperty("PRIVATE_HEADER"))
- {
- std::vector<std::string> relFiles;
- cmSystemTools::ExpandListArgument(files, relFiles);
- for(std::vector<std::string>::iterator it = relFiles.begin();
- it != relFiles.end(); ++it)
- {
- if(cmSourceFile* sf = this->Makefile->GetSource(*it))
- {
- SourceFileFlags& flags = this->SourceFlagsMap[sf];
- flags.MacFolder = "PrivateHeaders";
- flags.Type = cmGeneratorTarget::SourceFileTypePrivateHeader;
- }
- }
- }
- // Mark sources listed as resources.
- if(const char* files = this->Target->GetProperty("RESOURCE"))
- {
- std::vector<std::string> relFiles;
- cmSystemTools::ExpandListArgument(files, relFiles);
- for(std::vector<std::string>::iterator it = relFiles.begin();
- it != relFiles.end(); ++it)
- {
- if(cmSourceFile* sf = this->Makefile->GetSource(*it))
- {
- SourceFileFlags& flags = this->SourceFlagsMap[sf];
- flags.MacFolder = "Resources";
- flags.Type = cmGeneratorTarget::SourceFileTypeResource;
- }
- }
- }
- }
- //----------------------------------------------------------------------------
- const cmGeneratorTarget::CompatibleInterfacesBase&
- cmGeneratorTarget::GetCompatibleInterfaces(std::string const& config) const
- {
- cmGeneratorTarget::CompatibleInterfaces& compat =
- this->CompatibleInterfacesMap[config];
- if(!compat.Done)
- {
- compat.Done = true;
- compat.PropsBool.insert("POSITION_INDEPENDENT_CODE");
- compat.PropsString.insert("AUTOUIC_OPTIONS");
- std::vector<cmTarget const*> const& deps =
- this->GetLinkImplementationClosure(config);
- for(std::vector<cmTarget const*>::const_iterator li = deps.begin();
- li != deps.end(); ++li)
- {
- #define CM_READ_COMPATIBLE_INTERFACE(X, x) \
- if(const char* prop = (*li)->GetProperty("COMPATIBLE_INTERFACE_" #X)) \
- { \
- std::vector<std::string> props; \
- cmSystemTools::ExpandListArgument(prop, props); \
- compat.Props##x.insert(props.begin(), props.end()); \
- }
- CM_READ_COMPATIBLE_INTERFACE(BOOL, Bool)
- CM_READ_COMPATIBLE_INTERFACE(STRING, String)
- CM_READ_COMPATIBLE_INTERFACE(NUMBER_MIN, NumberMin)
- CM_READ_COMPATIBLE_INTERFACE(NUMBER_MAX, NumberMax)
- #undef CM_READ_COMPATIBLE_INTERFACE
- }
- }
- return compat;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::IsLinkInterfaceDependentBoolProperty(
- const std::string &p, const std::string& config) const
- {
- if (this->Target->GetType() == cmTarget::OBJECT_LIBRARY
- || this->Target->GetType() == cmTarget::INTERFACE_LIBRARY)
- {
- return false;
- }
- return this->GetCompatibleInterfaces(config).PropsBool.count(p) > 0;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::IsLinkInterfaceDependentStringProperty(
- const std::string &p, const std::string& config) const
- {
- if (this->Target->GetType() == cmTarget::OBJECT_LIBRARY
- || this->Target->GetType() == cmTarget::INTERFACE_LIBRARY)
- {
- return false;
- }
- return this->GetCompatibleInterfaces(config).PropsString.count(p) > 0;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::IsLinkInterfaceDependentNumberMinProperty(
- const std::string &p, const std::string& config) const
- {
- if (this->Target->GetType() == cmTarget::OBJECT_LIBRARY
- || this->Target->GetType() == cmTarget::INTERFACE_LIBRARY)
- {
- return false;
- }
- return this->GetCompatibleInterfaces(config).PropsNumberMin.count(p) > 0;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::IsLinkInterfaceDependentNumberMaxProperty(
- const std::string &p, const std::string& config) const
- {
- if (this->Target->GetType() == cmTarget::OBJECT_LIBRARY
- || this->Target->GetType() == cmTarget::INTERFACE_LIBRARY)
- {
- return false;
- }
- return this->GetCompatibleInterfaces(config).PropsNumberMax.count(p) > 0;
- }
- enum CompatibleType
- {
- BoolType,
- StringType,
- NumberMinType,
- NumberMaxType
- };
- template<typename PropertyType>
- PropertyType getLinkInterfaceDependentProperty(cmGeneratorTarget const* tgt,
- const std::string& prop,
- const std::string& config,
- CompatibleType,
- PropertyType *);
- template<>
- bool getLinkInterfaceDependentProperty(cmGeneratorTarget const* tgt,
- const std::string& prop,
- const std::string& config,
- CompatibleType, bool *)
- {
- return tgt->GetLinkInterfaceDependentBoolProperty(prop, config);
- }
- template<>
- const char * getLinkInterfaceDependentProperty(cmGeneratorTarget const* tgt,
- const std::string& prop,
- const std::string& config,
- CompatibleType t,
- const char **)
- {
- switch(t)
- {
- case BoolType:
- assert(0 && "String compatibility check function called for boolean");
- return 0;
- case StringType:
- return tgt->GetLinkInterfaceDependentStringProperty(prop, config);
- case NumberMinType:
- return tgt->GetLinkInterfaceDependentNumberMinProperty(prop, config);
- case NumberMaxType:
- return tgt->GetLinkInterfaceDependentNumberMaxProperty(prop, config);
- }
- assert(0 && "Unreachable!");
- return 0;
- }
- //----------------------------------------------------------------------------
- template<typename PropertyType>
- void checkPropertyConsistency(cmGeneratorTarget const* depender,
- cmTarget const* dependee,
- const std::string& propName,
- std::set<std::string> &emitted,
- const std::string& config,
- CompatibleType t,
- PropertyType *)
- {
- const char *prop = dependee->GetProperty(propName);
- if (!prop)
- {
- return;
- }
- std::vector<std::string> props;
- cmSystemTools::ExpandListArgument(prop, props);
- std::string pdir =
- dependee->GetMakefile()->GetRequiredDefinition("CMAKE_ROOT");
- pdir += "/Help/prop_tgt/";
- for(std::vector<std::string>::iterator pi = props.begin();
- pi != props.end(); ++pi)
- {
- std::string pname = cmSystemTools::HelpFileName(*pi);
- std::string pfile = pdir + pname + ".rst";
- if(cmSystemTools::FileExists(pfile.c_str(), true))
- {
- std::ostringstream e;
- e << "Target \"" << dependee->GetName() << "\" has property \""
- << *pi << "\" listed in its " << propName << " property. "
- "This is not allowed. Only user-defined properties may appear "
- "listed in the " << propName << " property.";
- depender->GetLocalGenerator()->IssueMessage(cmake::FATAL_ERROR, e.str());
- return;
- }
- if(emitted.insert(*pi).second)
- {
- getLinkInterfaceDependentProperty<PropertyType>(depender, *pi, config,
- t, 0);
- if (cmSystemTools::GetErrorOccuredFlag())
- {
- return;
- }
- }
- }
- }
- static std::string intersect(const std::set<std::string> &s1,
- const std::set<std::string> &s2)
- {
- std::set<std::string> intersect;
- std::set_intersection(s1.begin(),s1.end(),
- s2.begin(),s2.end(),
- std::inserter(intersect,intersect.begin()));
- if (!intersect.empty())
- {
- return *intersect.begin();
- }
- return "";
- }
- static std::string intersect(const std::set<std::string> &s1,
- const std::set<std::string> &s2,
- const std::set<std::string> &s3)
- {
- std::string result;
- result = intersect(s1, s2);
- if (!result.empty())
- return result;
- result = intersect(s1, s3);
- if (!result.empty())
- return result;
- return intersect(s2, s3);
- }
- static std::string intersect(const std::set<std::string> &s1,
- const std::set<std::string> &s2,
- const std::set<std::string> &s3,
- const std::set<std::string> &s4)
- {
- std::string result;
- result = intersect(s1, s2);
- if (!result.empty())
- return result;
- result = intersect(s1, s3);
- if (!result.empty())
- return result;
- result = intersect(s1, s4);
- if (!result.empty())
- return result;
- return intersect(s2, s3, s4);
- }
- //----------------------------------------------------------------------------
- void cmGeneratorTarget::CheckPropertyCompatibility(
- cmComputeLinkInformation *info, const std::string& config) const
- {
- const cmComputeLinkInformation::ItemVector &deps = info->GetItems();
- std::set<std::string> emittedBools;
- static std::string strBool = "COMPATIBLE_INTERFACE_BOOL";
- std::set<std::string> emittedStrings;
- static std::string strString = "COMPATIBLE_INTERFACE_STRING";
- std::set<std::string> emittedMinNumbers;
- static std::string strNumMin = "COMPATIBLE_INTERFACE_NUMBER_MIN";
- std::set<std::string> emittedMaxNumbers;
- static std::string strNumMax = "COMPATIBLE_INTERFACE_NUMBER_MAX";
- for(cmComputeLinkInformation::ItemVector::const_iterator li =
- deps.begin(); li != deps.end(); ++li)
- {
- if (!li->Target)
- {
- continue;
- }
- checkPropertyConsistency<bool>(this, li->Target,
- strBool,
- emittedBools, config, BoolType, 0);
- if (cmSystemTools::GetErrorOccuredFlag())
- {
- return;
- }
- checkPropertyConsistency<const char *>(this, li->Target,
- strString,
- emittedStrings, config,
- StringType, 0);
- if (cmSystemTools::GetErrorOccuredFlag())
- {
- return;
- }
- checkPropertyConsistency<const char *>(this, li->Target,
- strNumMin,
- emittedMinNumbers, config,
- NumberMinType, 0);
- if (cmSystemTools::GetErrorOccuredFlag())
- {
- return;
- }
- checkPropertyConsistency<const char *>(this, li->Target,
- strNumMax,
- emittedMaxNumbers, config,
- NumberMaxType, 0);
- if (cmSystemTools::GetErrorOccuredFlag())
- {
- return;
- }
- }
- std::string prop = intersect(emittedBools,
- emittedStrings,
- emittedMinNumbers,
- emittedMaxNumbers);
- if (!prop.empty())
- {
- // Use a sorted std::vector to keep the error message sorted.
- std::vector<std::string> props;
- std::set<std::string>::const_iterator i = emittedBools.find(prop);
- if (i != emittedBools.end())
- {
- props.push_back(strBool);
- }
- i = emittedStrings.find(prop);
- if (i != emittedStrings.end())
- {
- props.push_back(strString);
- }
- i = emittedMinNumbers.find(prop);
- if (i != emittedMinNumbers.end())
- {
- props.push_back(strNumMin);
- }
- i = emittedMaxNumbers.find(prop);
- if (i != emittedMaxNumbers.end())
- {
- props.push_back(strNumMax);
- }
- std::sort(props.begin(), props.end());
- std::string propsString = cmJoin(cmMakeRange(props).retreat(1), ", ");
- propsString += " and the " + props.back();
- std::ostringstream e;
- e << "Property \"" << prop << "\" appears in both the "
- << propsString <<
- " property in the dependencies of target \"" << this->GetName() <<
- "\". This is not allowed. A property may only require compatibility "
- "in a boolean interpretation, a numeric minimum, a numeric maximum or a "
- "string interpretation, but not a mixture.";
- this->LocalGenerator->IssueMessage(cmake::FATAL_ERROR, e.str());
- }
- }
- //----------------------------------------------------------------------------
- std::string compatibilityType(CompatibleType t)
- {
- switch(t)
- {
- case BoolType:
- return "Boolean compatibility";
- case StringType:
- return "String compatibility";
- case NumberMaxType:
- return "Numeric maximum compatibility";
- case NumberMinType:
- return "Numeric minimum compatibility";
- }
- assert(0 && "Unreachable!");
- return "";
- }
- //----------------------------------------------------------------------------
- std::string compatibilityAgree(CompatibleType t, bool dominant)
- {
- switch(t)
- {
- case BoolType:
- case StringType:
- return dominant ? "(Disagree)\n" : "(Agree)\n";
- case NumberMaxType:
- case NumberMinType:
- return dominant ? "(Dominant)\n" : "(Ignored)\n";
- }
- assert(0 && "Unreachable!");
- return "";
- }
- //----------------------------------------------------------------------------
- template<typename PropertyType>
- PropertyType getTypedProperty(cmTarget const* tgt, const std::string& prop);
- //----------------------------------------------------------------------------
- template<>
- bool getTypedProperty<bool>(cmTarget const* tgt, const std::string& prop)
- {
- return tgt->GetPropertyAsBool(prop);
- }
- //----------------------------------------------------------------------------
- template<>
- const char *getTypedProperty<const char *>(cmTarget const* tgt,
- const std::string& prop)
- {
- return tgt->GetProperty(prop);
- }
- template<typename PropertyType>
- std::string valueAsString(PropertyType);
- template<>
- std::string valueAsString<bool>(bool value)
- {
- return value ? "TRUE" : "FALSE";
- }
- template<>
- std::string valueAsString<const char*>(const char* value)
- {
- return value ? value : "(unset)";
- }
- template<typename PropertyType>
- PropertyType impliedValue(PropertyType);
- template<>
- bool impliedValue<bool>(bool)
- {
- return false;
- }
- template<>
- const char* impliedValue<const char*>(const char*)
- {
- return "";
- }
- //----------------------------------------------------------------------------
- template<typename PropertyType>
- std::pair<bool, PropertyType> consistentProperty(PropertyType lhs,
- PropertyType rhs,
- CompatibleType t);
- //----------------------------------------------------------------------------
- template<>
- std::pair<bool, bool> consistentProperty(bool lhs, bool rhs,
- CompatibleType)
- {
- return std::make_pair(lhs == rhs, lhs);
- }
- //----------------------------------------------------------------------------
- std::pair<bool, const char*> consistentStringProperty(const char *lhs,
- const char *rhs)
- {
- const bool b = strcmp(lhs, rhs) == 0;
- return std::make_pair(b, b ? lhs : 0);
- }
- //----------------------------------------------------------------------------
- std::pair<bool, const char*> consistentNumberProperty(const char *lhs,
- const char *rhs,
- CompatibleType t)
- {
- char *pEnd;
- const char* const null_ptr = 0;
- long lnum = strtol(lhs, &pEnd, 0);
- if (pEnd == lhs || *pEnd != '\0' || errno == ERANGE)
- {
- return std::pair<bool, const char*>(false, null_ptr);
- }
- long rnum = strtol(rhs, &pEnd, 0);
- if (pEnd == rhs || *pEnd != '\0' || errno == ERANGE)
- {
- return std::pair<bool, const char*>(false, null_ptr);
- }
- if (t == NumberMaxType)
- {
- return std::make_pair(true, std::max(lnum, rnum) == lnum ? lhs : rhs);
- }
- else
- {
- return std::make_pair(true, std::min(lnum, rnum) == lnum ? lhs : rhs);
- }
- }
- //----------------------------------------------------------------------------
- template<>
- std::pair<bool, const char*> consistentProperty(const char *lhs,
- const char *rhs,
- CompatibleType t)
- {
- if (!lhs && !rhs)
- {
- return std::make_pair(true, lhs);
- }
- if (!lhs)
- {
- return std::make_pair(true, rhs);
- }
- if (!rhs)
- {
- return std::make_pair(true, lhs);
- }
- const char* const null_ptr = 0;
- switch(t)
- {
- case BoolType:
- assert(0 && "consistentProperty for strings called with BoolType");
- return std::pair<bool, const char*>(false, null_ptr);
- case StringType:
- return consistentStringProperty(lhs, rhs);
- case NumberMinType:
- case NumberMaxType:
- return consistentNumberProperty(lhs, rhs, t);
- }
- assert(0 && "Unreachable!");
- return std::pair<bool, const char*>(false, null_ptr);
- }
- //----------------------------------------------------------------------------
- template<typename PropertyType>
- PropertyType checkInterfacePropertyCompatibility(cmGeneratorTarget const* tgt,
- const std::string &p,
- const std::string& config,
- const char *defaultValue,
- CompatibleType t,
- PropertyType *)
- {
- PropertyType propContent = getTypedProperty<PropertyType>(tgt->Target, p);
- const bool explicitlySet = tgt->Target->GetProperties()
- .find(p)
- != tgt->Target->GetProperties().end();
- const bool impliedByUse =
- tgt->Target->IsNullImpliedByLinkLibraries(p);
- assert((impliedByUse ^ explicitlySet)
- || (!impliedByUse && !explicitlySet));
- std::vector<cmTarget const*> const& deps =
- tgt->GetLinkImplementationClosure(config);
- if(deps.empty())
- {
- return propContent;
- }
- bool propInitialized = explicitlySet;
- std::string report = " * Target \"";
- report += tgt->GetName();
- if (explicitlySet)
- {
- report += "\" has property content \"";
- report += valueAsString<PropertyType>(propContent);
- report += "\"\n";
- }
- else if (impliedByUse)
- {
- report += "\" property is implied by use.\n";
- }
- else
- {
- report += "\" property not set.\n";
- }
- std::string interfaceProperty = "INTERFACE_" + p;
- for(std::vector<cmTarget const*>::const_iterator li =
- deps.begin();
- li != deps.end(); ++li)
- {
- // An error should be reported if one dependency
- // has INTERFACE_POSITION_INDEPENDENT_CODE ON and the other
- // has INTERFACE_POSITION_INDEPENDENT_CODE OFF, or if the
- // target itself has a POSITION_INDEPENDENT_CODE which disagrees
- // with a dependency.
- cmTarget const* theTarget = *li;
- const bool ifaceIsSet = theTarget->GetProperties()
- .find(interfaceProperty)
- != theTarget->GetProperties().end();
- PropertyType ifacePropContent =
- getTypedProperty<PropertyType>(theTarget,
- interfaceProperty);
- std::string reportEntry;
- if (ifaceIsSet)
- {
- reportEntry += " * Target \"";
- reportEntry += theTarget->GetName();
- reportEntry += "\" property value \"";
- reportEntry += valueAsString<PropertyType>(ifacePropContent);
- reportEntry += "\" ";
- }
- if (explicitlySet)
- {
- if (ifaceIsSet)
- {
- std::pair<bool, PropertyType> consistent =
- consistentProperty(propContent,
- ifacePropContent, t);
- report += reportEntry;
- report += compatibilityAgree(t, propContent != consistent.second);
- if (!consistent.first)
- {
- std::ostringstream e;
- e << "Property " << p << " on target \""
- << tgt->GetName() << "\" does\nnot match the "
- "INTERFACE_" << p << " property requirement\nof "
- "dependency \"" << theTarget->GetName() << "\".\n";
- cmSystemTools::Error(e.str().c_str());
- break;
- }
- else
- {
- propContent = consistent.second;
- continue;
- }
- }
- else
- {
- // Explicitly set on target and not set in iface. Can't disagree.
- continue;
- }
- }
- else if (impliedByUse)
- {
- propContent = impliedValue<PropertyType>(propContent);
- if (ifaceIsSet)
- {
- std::pair<bool, PropertyType> consistent =
- consistentProperty(propContent,
- ifacePropContent, t);
- report += reportEntry;
- report += compatibilityAgree(t, propContent != consistent.second);
- if (!consistent.first)
- {
- std::ostringstream e;
- e << "Property " << p << " on target \""
- << tgt->GetName() << "\" is\nimplied to be " << defaultValue
- << " because it was used to determine the link libraries\n"
- "already. The INTERFACE_" << p << " property on\ndependency \""
- << theTarget->GetName() << "\" is in conflict.\n";
- cmSystemTools::Error(e.str().c_str());
- break;
- }
- else
- {
- propContent = consistent.second;
- continue;
- }
- }
- else
- {
- // Implicitly set on target and not set in iface. Can't disagree.
- continue;
- }
- }
- else
- {
- if (ifaceIsSet)
- {
- if (propInitialized)
- {
- std::pair<bool, PropertyType> consistent =
- consistentProperty(propContent,
- ifacePropContent, t);
- report += reportEntry;
- report += compatibilityAgree(t, propContent != consistent.second);
- if (!consistent.first)
- {
- std::ostringstream e;
- e << "The INTERFACE_" << p << " property of \""
- << theTarget->GetName() << "\" does\nnot agree with the value "
- "of " << p << " already determined\nfor \""
- << tgt->GetName() << "\".\n";
- cmSystemTools::Error(e.str().c_str());
- break;
- }
- else
- {
- propContent = consistent.second;
- continue;
- }
- }
- else
- {
- report += reportEntry + "(Interface set)\n";
- propContent = ifacePropContent;
- propInitialized = true;
- }
- }
- else
- {
- // Not set. Nothing to agree on.
- continue;
- }
- }
- }
- tgt->ReportPropertyOrigin(p, valueAsString<PropertyType>(propContent),
- report, compatibilityType(t));
- return propContent;
- }
- //----------------------------------------------------------------------------
- bool cmGeneratorTarget::GetLinkInterfaceDependentBoolProperty(
- const std::string &p, const std::string& config) const
- {
- return checkInterfacePropertyCompatibility<bool>(this, p, config,
- "FALSE",
- BoolType, 0);
- }
- //----------------------------------------------------------------------------
- const char* cmGeneratorTarget::GetLinkInterfaceDependentStringProperty(
- const std::string &p,
- const std::string& config) const
- {
- return checkInterfacePropertyCompatibility<const char *>(this,
- p,
- config,
- "empty",
- StringType, 0);
- }
- //----------------------------------------------------------------------------
- const char * cmGeneratorTarget::GetLinkInterfaceDependentNumberMinProperty(
- const std::string &p,
- const std::string& config) const
- {
- return checkInterfacePropertyCompatibility<const char *>(this,
- p,
- config,
- "empty",
- NumberMinType, 0);
- }
- //----------------------------------------------------------------------------
- const char * cmGeneratorTarget::GetLinkInterfaceDependentNumberMaxProperty(
- const std::string &p,
- const std::string& config) const
- {
- return checkInterfacePropertyCompatibility<const char *>(this,
- p,
- config,
- "empty",
- NumberMaxType, 0);
- }
- //----------------------------------------------------------------------------
- cmComputeLinkInformation*
- cmGeneratorTarget::GetLinkInformation(const std::string& config) const
- {
- // Lookup any existing information for this configuration.
- std::string key(cmSystemTools::UpperCase(config));
- cmTargetLinkInformationMap::iterator
- i = this->LinkInformation.find(key);
- if(i == this->LinkInformation.end())
- {
- // Compute information for this configuration.
- cmComputeLinkInformation* info =
- new cmComputeLinkInformation(this->Target, config);
- if(!info || !info->Compute())
- {
- delete info;
- info = 0;
- }
- // Store the information for this configuration.
- cmTargetLinkInformationMap::value_type entry(key, info);
- i = this->LinkInformation.insert(entry).first;
- if (info)
- {
- this->CheckPropertyCompatibility(info, config);
- }
- }
- return i->second;
- }
- //----------------------------------------------------------------------------
- void
- cmGeneratorTarget::ReportPropertyOrigin(const std::string &p,
- const std::string &result,
- const std::string &report,
- const std::string &compatibilityType) const
- {
- std::vector<std::string> debugProperties;
- const char *debugProp = this->Target->GetMakefile()
- ->GetDefinition("CMAKE_DEBUG_TARGET_PROPERTIES");
- if (debugProp)
- {
- cmSystemTools::ExpandListArgument(debugProp, debugProperties);
- }
- bool debugOrigin = !this->DebugCompatiblePropertiesDone[p]
- && std::find(debugProperties.begin(),
- debugProperties.end(),
- p)
- != debugProperties.end();
- if (this->Target->GetMakefile()->IsConfigured())
- {
- this->DebugCompatiblePropertiesDone[p] = true;
- }
- if (!debugOrigin)
- {
- return;
- }
- std::string areport = compatibilityType;
- areport += std::string(" of property \"") + p + "\" for target \"";
- areport += std::string(this->GetName());
- areport += "\" (result: \"";
- areport += result;
- areport += "\"):\n" + report;
- this->Makefile->GetCMakeInstance()->IssueMessage(cmake::LOG, areport);
- }
|