ENH: Added support for element tag specification with syntax tag:element as an argument to the CABLE_DEFINE_SET command. A single colon with nothing to its left will result in an empty tag.

This commit is contained in:
Brad King 2001-03-13 18:01:49 -05:00
parent 50c9139628
commit 007f5ccd5f
2 changed files with 73 additions and 19 deletions

View File

@ -39,7 +39,10 @@ bool cmCableDefineSetCommand::Invoke(std::vector<std::string>& args)
// The rest of the arguments are the elements to be placed in the set. // The rest of the arguments are the elements to be placed in the set.
for(; arg != args.end(); ++arg) for(; arg != args.end(); ++arg)
{ {
m_Elements.push_back(Element(this->GenerateTag(*arg), *arg)); // If the element cannot be added, return an error.
// This can occur when a tag is not specified and can't be generated.
if(!this->AddElement(*arg))
{ return false; }
} }
// Write this command's configuration output. // Write this command's configuration output.
@ -86,6 +89,43 @@ void cmCableDefineSetCommand::WriteConfiguration() const
} }
/**
* Add an element to the set. The given string is the argument to the
* command describing the element. There are two formats allowed:
* "code" = The code describing the element to CABLE is simply given.
* The GenerateTag() method will guess at a good tag for the
* code.
* "tag:code" = The left side of a single colon is text describing the tag.
* GenerateTag() will not be called.
*/
bool cmCableDefineSetCommand::AddElement(const std::string& arg)
{
// A regular expression to match the tagged element specification.
cmRegularExpression tagGiven("^([A-Za-z_0-9]*)[ \t]*:[ \t]*([^:].*|::.*)$");
std::string tag;
std::string code;
if(tagGiven.find(arg.c_str()))
{
// A tag was given. Use it.
tag = tagGiven.match(1);
code = tagGiven.match(2);
}
else
{
// No tag was given. Try to generate one.
if(!this->GenerateTag(arg, tag))
{ return false; }
code = arg;
}
// Add an element with the given tag and code.
m_Elements.push_back(Element(tag, code));
return true;
}
/** /**
* Given the string representing a set element, automatically generate * Given the string representing a set element, automatically generate
* the CABLE element tag for it. * the CABLE element tag for it.
@ -93,8 +133,9 @@ void cmCableDefineSetCommand::WriteConfiguration() const
* **This function determines how the output language of all * **This function determines how the output language of all
* CABLE-generated wrappers will look!** * CABLE-generated wrappers will look!**
*/ */
std::string bool
cmCableDefineSetCommand::GenerateTag(const std::string& element) const cmCableDefineSetCommand::GenerateTag(const std::string& element,
std::string& tag)
{ {
// Hold the regular expressions for matching against the element. // Hold the regular expressions for matching against the element.
cmRegularExpression regex; cmRegularExpression regex;
@ -103,47 +144,48 @@ cmCableDefineSetCommand::GenerateTag(const std::string& element) const
// The set's elements have their own tags, so we don't need one. // The set's elements have their own tags, so we don't need one.
regex.compile("^[ \t]*\\$"); regex.compile("^[ \t]*\\$");
if(regex.find(element)) if(regex.find(element))
{ return ""; } { tag = ""; return true; }
// Test for simple integer // Test for simple integer
regex.compile("^[ \t]*([0-9]*)[ \t]*$"); regex.compile("^[ \t]*([0-9]*)[ \t]*$");
if(regex.find(element)) if(regex.find(element))
{ {
std::string tag = "_"; tag = "_";
tag.append(regex.match(1)); tag.append(regex.match(1));
return tag; return true;
} }
// Test for basic integer type // Test for basic integer type
regex.compile("^[ \t]*(unsigned[ ]|signed[ ])?[ \t]*(char|short|int|long|long[ ]long)[ \t]*$"); regex.compile("^[ \t]*(unsigned[ ]|signed[ ])?[ \t]*(char|short|int|long|long[ ]long)[ \t]*$");
if(regex.find(element)) if(regex.find(element))
{ {
std::string tag = "_"; tag = "_";
if(regex.match(1) == "unsigned ") if(regex.match(1) == "unsigned ")
{ tag.append("u"); } { tag.append("u"); }
if(regex.match(2) == "long long") if(regex.match(2) == "long long")
{ tag.append("llong"); } { tag.append("llong"); }
else else
{ tag.append(regex.match(2)); } { tag.append(regex.match(2)); }
return tag; return true;
} }
// Test for basic floating-point type // Test for basic floating-point type
regex.compile("^[ \t]*(long[ ])?[ \t]*(float|double)[ \t]*$"); regex.compile("^[ \t]*(long[ ])?[ \t]*(float|double)[ \t]*$");
if(regex.find(element)) if(regex.find(element))
{ {
std::string tag = "_"; tag = "_";
if(regex.match(1) == "long ") if(regex.match(1) == "long ")
tag.append("l"); tag.append("l");
tag.append(regex.match(2)); tag.append(regex.match(2));
return tag; return true;
} }
// Test for basic wide-character type // Test for basic wide-character type
regex.compile("^[ \t]*(wchar_t)[ \t]*$"); regex.compile("^[ \t]*(wchar_t)[ \t]*$");
if(regex.find(element)) if(regex.find(element))
{ {
return "_wchar"; tag = "_wchar";
return true;
} }
// Test for plain type name (without template arguments). // Test for plain type name (without template arguments).
@ -151,7 +193,8 @@ cmCableDefineSetCommand::GenerateTag(const std::string& element) const
if(regex.find(element)) if(regex.find(element))
{ {
// The tag is the same as the type. // The tag is the same as the type.
return regex.match(1); tag = regex.match(1);
return true;
} }
// Test for template class instance. // Test for template class instance.
@ -160,8 +203,17 @@ cmCableDefineSetCommand::GenerateTag(const std::string& element) const
{ {
// The tag is the type without arguments (the arguments may have // The tag is the type without arguments (the arguments may have
// their own tags). // their own tags).
return regex.match(1); tag = regex.match(1);
return true;
} }
return "NO_AUTO_TAG"; // We can't generate a tag.
std::string err =
("doesn't know how to generate tag for element \""+element+"\" in set \""
+m_SetName+"\"\nPlease specify one with the \"tag:element\" syntax.");
this->SetError(err.c_str());
tag = "";
return false;
} }

View File

@ -69,17 +69,19 @@ public:
virtual const char* GetFullDocumentation() virtual const char* GetFullDocumentation()
{ {
return return
"CABLE_DEFINE_SET(name_of_set member1 member2 ...)\n" "CABLE_DEFINE_SET(name_of_set [[tag1]:]memeber1 [[tag2]:]member2 ...)\n"
"Generates a Set definition in the CABLE configuration. Tags are\n" "Generates a Set definition in the CABLE configuration. The sets are\n"
"automatically generated. The sets are referenced in other CABLE\n" "referenced in other CABLE commands by a '$' immediately followed by\n"
"commands by a '$' immediately followed by the set name (ex. $SetName)."; "the set name (ex. $SetName). If a the \"tag:\" syntax is not used,\n"
"an attempt is made to auto-generate a meaningful tag.\n";
} }
cmTypeMacro(cmCableDefineSetCommand, cmCableCommand); cmTypeMacro(cmCableDefineSetCommand, cmCableCommand);
private: private:
void WriteConfiguration() const; void WriteConfiguration() const;
std::string GenerateTag(const std::string&) const; bool AddElement(const std::string&);
bool GenerateTag(const std::string&, std::string&);
private: private:
typedef std::pair<std::string, std::string> Element; typedef std::pair<std::string, std::string> Element;
typedef std::vector<Element> Elements; typedef std::vector<Element> Elements;