Add OBJECT_LIBRARY target type
This library type can compile sources to object files but does not link or archive them. It will be useful to reference from executable and normal library targets for direct inclusion of object files in them. Diagnose and reject the following as errors: * An OBJECT library may not be referenced in target_link_libraries. * An OBJECT library may contain only compiling sources and supporting headers and custom commands. Other source types that are not normally ignored are not allowed. * An OBJECT library may not have PRE_BUILD, PRE_LINK, or POST_BUILD commands. * An OBJECT library may not be installed, exported, or imported. Some of these cases may be supported in the future but are not for now. Teach the VS generator that OBJECT_LIBRARY targets are "linkable" just like STATIC_LIBRARY targets for the LinkLibraryDependencies behavior.
This commit is contained in:
parent
d5aedf15a4
commit
b87d7a60a0
|
@ -64,6 +64,12 @@ bool cmAddLibraryCommand
|
||||||
type = cmTarget::MODULE_LIBRARY;
|
type = cmTarget::MODULE_LIBRARY;
|
||||||
haveSpecifiedType = true;
|
haveSpecifiedType = true;
|
||||||
}
|
}
|
||||||
|
else if(libType == "OBJECT")
|
||||||
|
{
|
||||||
|
++s;
|
||||||
|
type = cmTarget::OBJECT_LIBRARY;
|
||||||
|
haveSpecifiedType = true;
|
||||||
|
}
|
||||||
else if(libType == "UNKNOWN")
|
else if(libType == "UNKNOWN")
|
||||||
{
|
{
|
||||||
++s;
|
++s;
|
||||||
|
@ -118,6 +124,14 @@ bool cmAddLibraryCommand
|
||||||
this->SetError("called with IMPORTED argument but no library type.");
|
this->SetError("called with IMPORTED argument but no library type.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
if(type == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
this->Makefile->IssueMessage(
|
||||||
|
cmake::FATAL_ERROR,
|
||||||
|
"The OBJECT library type may not be used for IMPORTED libraries."
|
||||||
|
);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Make sure the target does not already exist.
|
// Make sure the target does not already exist.
|
||||||
if(this->Makefile->FindTargetToUse(libName.c_str()))
|
if(this->Makefile->FindTargetToUse(libName.c_str()))
|
||||||
|
|
|
@ -633,6 +633,19 @@ cmTarget* cmComputeLinkDepends::FindTargetToLink(int depender_index,
|
||||||
tgt = 0;
|
tgt = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(tgt && tgt->GetType() == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "Target \"" << this->Target->GetName() << "\" links to "
|
||||||
|
"OBJECT library \"" << tgt->GetName() << "\" but this is not "
|
||||||
|
"allowed. "
|
||||||
|
"One may link only to STATIC or SHARED libraries, or to executables "
|
||||||
|
"with the ENABLE_EXPORTS property set.";
|
||||||
|
this->CMakeInstance->IssueMessage(cmake::FATAL_ERROR, e.str(),
|
||||||
|
this->Target->GetBacktrace());
|
||||||
|
tgt = 0;
|
||||||
|
}
|
||||||
|
|
||||||
// Return the target found, if any.
|
// Return the target found, if any.
|
||||||
return tgt;
|
return tgt;
|
||||||
}
|
}
|
||||||
|
|
|
@ -124,6 +124,14 @@ bool cmExportCommand
|
||||||
{
|
{
|
||||||
targets.push_back(target);
|
targets.push_back(target);
|
||||||
}
|
}
|
||||||
|
else if(target->GetType() == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "given OBJECT library \"" << *currentTarget
|
||||||
|
<< "\" which may not be exported.";
|
||||||
|
this->SetError(e.str().c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
cmOStringStream e;
|
cmOStringStream e;
|
||||||
|
|
|
@ -29,6 +29,8 @@ cmGeneratorTarget::cmGeneratorTarget(cmTarget* t): Target(t)
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
void cmGeneratorTarget::ClassifySources()
|
void cmGeneratorTarget::ClassifySources()
|
||||||
{
|
{
|
||||||
|
bool isObjLib = this->Target->GetType() == cmTarget::OBJECT_LIBRARY;
|
||||||
|
std::vector<cmSourceFile*> badObjLib;
|
||||||
std::vector<cmSourceFile*> const& sources = this->Target->GetSourceFiles();
|
std::vector<cmSourceFile*> const& sources = this->Target->GetSourceFiles();
|
||||||
for(std::vector<cmSourceFile*>::const_iterator si = sources.begin();
|
for(std::vector<cmSourceFile*>::const_iterator si = sources.begin();
|
||||||
si != sources.end(); ++si)
|
si != sources.end(); ++si)
|
||||||
|
@ -43,6 +45,7 @@ void cmGeneratorTarget::ClassifySources()
|
||||||
else if(tsFlags.Type != cmTarget::SourceFileTypeNormal)
|
else if(tsFlags.Type != cmTarget::SourceFileTypeNormal)
|
||||||
{
|
{
|
||||||
this->OSXContent.push_back(sf);
|
this->OSXContent.push_back(sf);
|
||||||
|
if(isObjLib) { badObjLib.push_back(sf); }
|
||||||
}
|
}
|
||||||
else if(sf->GetPropertyAsBool("HEADER_FILE_ONLY"))
|
else if(sf->GetPropertyAsBool("HEADER_FILE_ONLY"))
|
||||||
{
|
{
|
||||||
|
@ -51,10 +54,12 @@ void cmGeneratorTarget::ClassifySources()
|
||||||
else if(sf->GetPropertyAsBool("EXTERNAL_OBJECT"))
|
else if(sf->GetPropertyAsBool("EXTERNAL_OBJECT"))
|
||||||
{
|
{
|
||||||
this->ExternalObjects.push_back(sf);
|
this->ExternalObjects.push_back(sf);
|
||||||
|
if(isObjLib) { badObjLib.push_back(sf); }
|
||||||
}
|
}
|
||||||
else if(cmSystemTools::LowerCase(sf->GetExtension()) == "def")
|
else if(cmSystemTools::LowerCase(sf->GetExtension()) == "def")
|
||||||
{
|
{
|
||||||
this->ModuleDefinitionFile = sf->GetFullPath();
|
this->ModuleDefinitionFile = sf->GetFullPath();
|
||||||
|
if(isObjLib) { badObjLib.push_back(sf); }
|
||||||
}
|
}
|
||||||
else if(this->GlobalGenerator->IgnoreFile(sf->GetExtension().c_str()))
|
else if(this->GlobalGenerator->IgnoreFile(sf->GetExtension().c_str()))
|
||||||
{
|
{
|
||||||
|
@ -69,6 +74,22 @@ void cmGeneratorTarget::ClassifySources()
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
this->ExtraSources.push_back(sf);
|
this->ExtraSources.push_back(sf);
|
||||||
|
if(isObjLib) { badObjLib.push_back(sf); }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(!badObjLib.empty())
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "OBJECT library \"" << this->Target->GetName() << "\" contains:\n";
|
||||||
|
for(std::vector<cmSourceFile*>::iterator i = badObjLib.begin();
|
||||||
|
i != badObjLib.end(); ++i)
|
||||||
|
{
|
||||||
|
e << " " << (*i)->GetLocation().GetName() << "\n";
|
||||||
|
}
|
||||||
|
e << "but may contain only headers and sources that compile.";
|
||||||
|
this->GlobalGenerator->GetCMakeInstance()
|
||||||
|
->IssueMessage(cmake::FATAL_ERROR, e.str(),
|
||||||
|
this->Target->GetBacktrace());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -374,6 +374,12 @@ bool cmGlobalVisualStudioGenerator::ComputeTargetDepends()
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
static bool VSLinkable(cmTarget* t)
|
||||||
|
{
|
||||||
|
return t->IsLinkable() || t->GetType() == cmTarget::OBJECT_LIBRARY;
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
void cmGlobalVisualStudioGenerator::ComputeVSTargetDepends(cmTarget& target)
|
void cmGlobalVisualStudioGenerator::ComputeVSTargetDepends(cmTarget& target)
|
||||||
{
|
{
|
||||||
|
@ -458,7 +464,7 @@ void cmGlobalVisualStudioGenerator::ComputeVSTargetDepends(cmTarget& target)
|
||||||
di != utilDepends.end(); ++di)
|
di != utilDepends.end(); ++di)
|
||||||
{
|
{
|
||||||
cmTarget* dep = *di;
|
cmTarget* dep = *di;
|
||||||
if(allowLinkable || !dep->IsLinkable() || linked.count(dep))
|
if(allowLinkable || !VSLinkable(dep) || linked.count(dep))
|
||||||
{
|
{
|
||||||
// Direct dependency allowed.
|
// Direct dependency allowed.
|
||||||
vsTargetDepend.insert(dep->GetName());
|
vsTargetDepend.insert(dep->GetName());
|
||||||
|
|
|
@ -357,7 +357,8 @@ bool cmInstallCommand::HandleTargetsMode(std::vector<std::string> const& args)
|
||||||
if(target->GetType() != cmTarget::EXECUTABLE &&
|
if(target->GetType() != cmTarget::EXECUTABLE &&
|
||||||
target->GetType() != cmTarget::STATIC_LIBRARY &&
|
target->GetType() != cmTarget::STATIC_LIBRARY &&
|
||||||
target->GetType() != cmTarget::SHARED_LIBRARY &&
|
target->GetType() != cmTarget::SHARED_LIBRARY &&
|
||||||
target->GetType() != cmTarget::MODULE_LIBRARY)
|
target->GetType() != cmTarget::MODULE_LIBRARY &&
|
||||||
|
target->GetType() != cmTarget::OBJECT_LIBRARY)
|
||||||
{
|
{
|
||||||
cmOStringStream e;
|
cmOStringStream e;
|
||||||
e << "TARGETS given target \"" << (*targetIt)
|
e << "TARGETS given target \"" << (*targetIt)
|
||||||
|
@ -365,6 +366,14 @@ bool cmInstallCommand::HandleTargetsMode(std::vector<std::string> const& args)
|
||||||
this->SetError(e.str().c_str());
|
this->SetError(e.str().c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
else if(target->GetType() == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "TARGETS given OBJECT library \"" << (*targetIt)
|
||||||
|
<< "\" which may not be installed.";
|
||||||
|
this->SetError(e.str().c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
// Store the target in the list to be installed.
|
// Store the target in the list to be installed.
|
||||||
targets.push_back(target);
|
targets.push_back(target);
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,6 +90,7 @@ void cmInstallTargetGenerator::GenerateScriptForConfig(std::ostream& os,
|
||||||
case cmTarget::STATIC_LIBRARY: type = cmInstallType_STATIC_LIBRARY; break;
|
case cmTarget::STATIC_LIBRARY: type = cmInstallType_STATIC_LIBRARY; break;
|
||||||
case cmTarget::SHARED_LIBRARY: type = cmInstallType_SHARED_LIBRARY; break;
|
case cmTarget::SHARED_LIBRARY: type = cmInstallType_SHARED_LIBRARY; break;
|
||||||
case cmTarget::MODULE_LIBRARY: type = cmInstallType_MODULE_LIBRARY; break;
|
case cmTarget::MODULE_LIBRARY: type = cmInstallType_MODULE_LIBRARY; break;
|
||||||
|
case cmTarget::OBJECT_LIBRARY:
|
||||||
case cmTarget::UTILITY:
|
case cmTarget::UTILITY:
|
||||||
case cmTarget::GLOBAL_TARGET:
|
case cmTarget::GLOBAL_TARGET:
|
||||||
case cmTarget::UNKNOWN_LIBRARY:
|
case cmTarget::UNKNOWN_LIBRARY:
|
||||||
|
|
|
@ -1904,6 +1904,10 @@ bool cmLocalGenerator::GetRealDependency(const char* inName,
|
||||||
case cmTarget::UNKNOWN_LIBRARY:
|
case cmTarget::UNKNOWN_LIBRARY:
|
||||||
dep = target->GetLocation(config);
|
dep = target->GetLocation(config);
|
||||||
return true;
|
return true;
|
||||||
|
case cmTarget::OBJECT_LIBRARY:
|
||||||
|
// An object library has no single file on which to depend.
|
||||||
|
// This was listed to get the target-level dependency.
|
||||||
|
return false;
|
||||||
case cmTarget::UTILITY:
|
case cmTarget::UTILITY:
|
||||||
case cmTarget::GLOBAL_TARGET:
|
case cmTarget::GLOBAL_TARGET:
|
||||||
// A utility target has no file on which to depend. This was listed
|
// A utility target has no file on which to depend. This was listed
|
||||||
|
|
|
@ -853,6 +853,14 @@ cmMakefile::AddCustomCommandToTarget(const char* target,
|
||||||
cmTargets::iterator ti = this->Targets.find(target);
|
cmTargets::iterator ti = this->Targets.find(target);
|
||||||
if(ti != this->Targets.end())
|
if(ti != this->Targets.end())
|
||||||
{
|
{
|
||||||
|
if(ti->second.GetType() == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "Target \"" << target << "\" is an OBJECT library "
|
||||||
|
"that may not have PRE_BUILD, PRE_LINK, or POST_BUILD commands.";
|
||||||
|
this->IssueMessage(cmake::FATAL_ERROR, e.str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
// Add the command to the appropriate build step for the target.
|
// Add the command to the appropriate build step for the target.
|
||||||
std::vector<std::string> no_output;
|
std::vector<std::string> no_output;
|
||||||
cmCustomCommand cc(this, no_output, depends,
|
cmCustomCommand cc(this, no_output, depends,
|
||||||
|
@ -1912,8 +1920,11 @@ cmTarget* cmMakefile::AddLibrary(const char* lname, cmTarget::TargetType type,
|
||||||
// wrong type ? default to STATIC
|
// wrong type ? default to STATIC
|
||||||
if ( (type != cmTarget::STATIC_LIBRARY)
|
if ( (type != cmTarget::STATIC_LIBRARY)
|
||||||
&& (type != cmTarget::SHARED_LIBRARY)
|
&& (type != cmTarget::SHARED_LIBRARY)
|
||||||
&& (type != cmTarget::MODULE_LIBRARY))
|
&& (type != cmTarget::MODULE_LIBRARY)
|
||||||
|
&& (type != cmTarget::OBJECT_LIBRARY))
|
||||||
{
|
{
|
||||||
|
this->IssueMessage(cmake::INTERNAL_ERROR,
|
||||||
|
"cmMakefile::AddLibrary given invalid target type.");
|
||||||
type = cmTarget::STATIC_LIBRARY;
|
type = cmTarget::STATIC_LIBRARY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,8 @@ const char* cmTarget::GetTargetTypeName(TargetType targetType)
|
||||||
return "MODULE_LIBRARY";
|
return "MODULE_LIBRARY";
|
||||||
case cmTarget::SHARED_LIBRARY:
|
case cmTarget::SHARED_LIBRARY:
|
||||||
return "SHARED_LIBRARY";
|
return "SHARED_LIBRARY";
|
||||||
|
case cmTarget::OBJECT_LIBRARY:
|
||||||
|
return "OBJECT_LIBRARY";
|
||||||
case cmTarget::EXECUTABLE:
|
case cmTarget::EXECUTABLE:
|
||||||
return "EXECUTABLE";
|
return "EXECUTABLE";
|
||||||
case cmTarget::UTILITY:
|
case cmTarget::UTILITY:
|
||||||
|
|
|
@ -59,7 +59,8 @@ class cmTarget
|
||||||
public:
|
public:
|
||||||
cmTarget();
|
cmTarget();
|
||||||
enum TargetType { EXECUTABLE, STATIC_LIBRARY,
|
enum TargetType { EXECUTABLE, STATIC_LIBRARY,
|
||||||
SHARED_LIBRARY, MODULE_LIBRARY, UTILITY, GLOBAL_TARGET,
|
SHARED_LIBRARY, MODULE_LIBRARY,
|
||||||
|
OBJECT_LIBRARY, UTILITY, GLOBAL_TARGET,
|
||||||
UNKNOWN_LIBRARY};
|
UNKNOWN_LIBRARY};
|
||||||
static const char* GetTargetTypeName(TargetType targetType);
|
static const char* GetTargetTypeName(TargetType targetType);
|
||||||
enum CustomCommandType { PRE_BUILD, PRE_LINK, POST_BUILD };
|
enum CustomCommandType { PRE_BUILD, PRE_LINK, POST_BUILD };
|
||||||
|
|
|
@ -84,6 +84,16 @@ bool cmTargetLinkLibrariesCommand
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(this->Target->GetType() == cmTarget::OBJECT_LIBRARY)
|
||||||
|
{
|
||||||
|
cmOStringStream e;
|
||||||
|
e << "Object library target \"" << args[0] << "\" "
|
||||||
|
<< "may not link to anything.";
|
||||||
|
this->Makefile->IssueMessage(cmake::FATAL_ERROR, e.str());
|
||||||
|
cmSystemTools::SetFatalErrorOccured();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// but we might not have any libs after variable expansion
|
// but we might not have any libs after variable expansion
|
||||||
if(args.size() < 2)
|
if(args.size() < 2)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue