xref: /aoo4110/main/xml2cmp/source/xcd/parse.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski 
25*b1cdbd2cSJim Jagielski #include <parse.hxx>
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include <string.h>
28*b1cdbd2cSJim Jagielski #include <iostream>
29*b1cdbd2cSJim Jagielski #include <xmlelem.hxx>
30*b1cdbd2cSJim Jagielski 
31*b1cdbd2cSJim Jagielski #if (_MSC_VER >=1400)
32*b1cdbd2cSJim Jagielski #pragma warning(disable:4365)
33*b1cdbd2cSJim Jagielski #endif
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski #ifdef UNX
36*b1cdbd2cSJim Jagielski #define strnicmp strncasecmp
37*b1cdbd2cSJim Jagielski #endif
38*b1cdbd2cSJim Jagielski 
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski 
41*b1cdbd2cSJim Jagielski // NOT FULLY DEFINED SERVICES
42*b1cdbd2cSJim Jagielski 
43*b1cdbd2cSJim Jagielski 
44*b1cdbd2cSJim Jagielski 
45*b1cdbd2cSJim Jagielski #define AssertionOf(x)	\
46*b1cdbd2cSJim Jagielski 	{if (!(x)) {std::cerr << "Assertion failed: " << #x << __FILE__ << __LINE__ << std::endl; exit(3); }}
47*b1cdbd2cSJim Jagielski 
48*b1cdbd2cSJim Jagielski 
49*b1cdbd2cSJim Jagielski 
X2CParser(XmlElement & o_rDocumentData)50*b1cdbd2cSJim Jagielski X2CParser::X2CParser( XmlElement & o_rDocumentData )
51*b1cdbd2cSJim Jagielski 	:	// sFileName,
52*b1cdbd2cSJim Jagielski 		nFileLine(0),
53*b1cdbd2cSJim Jagielski 		pDocumentData(&o_rDocumentData),
54*b1cdbd2cSJim Jagielski         // sWord,
55*b1cdbd2cSJim Jagielski 		text(0)
56*b1cdbd2cSJim Jagielski {
57*b1cdbd2cSJim Jagielski }
58*b1cdbd2cSJim Jagielski 
~X2CParser()59*b1cdbd2cSJim Jagielski X2CParser::~X2CParser()
60*b1cdbd2cSJim Jagielski {
61*b1cdbd2cSJim Jagielski }
62*b1cdbd2cSJim Jagielski 
63*b1cdbd2cSJim Jagielski 
64*b1cdbd2cSJim Jagielski bool
LoadFile(const char * i_sFilename)65*b1cdbd2cSJim Jagielski X2CParser::LoadFile( const char * i_sFilename )
66*b1cdbd2cSJim Jagielski {
67*b1cdbd2cSJim Jagielski 	sFileName = i_sFilename;
68*b1cdbd2cSJim Jagielski 	nFileLine = 1;
69*b1cdbd2cSJim Jagielski 
70*b1cdbd2cSJim Jagielski 	// Load file:
71*b1cdbd2cSJim Jagielski 	if ( ! LoadXmlFile( aFile, i_sFilename ) )
72*b1cdbd2cSJim Jagielski 		return false;
73*b1cdbd2cSJim Jagielski 
74*b1cdbd2cSJim Jagielski 	// Test correct end:
75*b1cdbd2cSJim Jagielski 	const char * pLastTag = strrchr(aFile.operator const char *(),'<');
76*b1cdbd2cSJim Jagielski 	if (pLastTag == 0)
77*b1cdbd2cSJim Jagielski 		return false;
78*b1cdbd2cSJim Jagielski 	if ( strnicmp(pLastTag+2, pDocumentData->Name().str(), pDocumentData->Name().l()) != 0
79*b1cdbd2cSJim Jagielski 		 || strnicmp(pLastTag, "</", 2) != 0 )
80*b1cdbd2cSJim Jagielski 		return false;
81*b1cdbd2cSJim Jagielski 	if (strchr(pLastTag,'>') == 0)
82*b1cdbd2cSJim Jagielski 		return false;
83*b1cdbd2cSJim Jagielski 	return true;
84*b1cdbd2cSJim Jagielski }
85*b1cdbd2cSJim Jagielski 
86*b1cdbd2cSJim Jagielski void
Parse()87*b1cdbd2cSJim Jagielski X2CParser::Parse()
88*b1cdbd2cSJim Jagielski {
89*b1cdbd2cSJim Jagielski 	// Parse:
90*b1cdbd2cSJim Jagielski 	text = aFile.operator const char *();
91*b1cdbd2cSJim Jagielski 
92*b1cdbd2cSJim Jagielski 	Parse_XmlDeclaration();
93*b1cdbd2cSJim Jagielski 	Parse_Doctype();
94*b1cdbd2cSJim Jagielski 
95*b1cdbd2cSJim Jagielski     // skip XML comment
96*b1cdbd2cSJim Jagielski     Goto('<');
97*b1cdbd2cSJim Jagielski     if ( IsText("<!--") )
98*b1cdbd2cSJim Jagielski         Goto_And_Pass('>');
99*b1cdbd2cSJim Jagielski 
100*b1cdbd2cSJim Jagielski 	pDocumentData->Parse(*this);
101*b1cdbd2cSJim Jagielski }
102*b1cdbd2cSJim Jagielski 
103*b1cdbd2cSJim Jagielski bool
Parse(const char * i_sFilename)104*b1cdbd2cSJim Jagielski X2CParser::Parse( const char * i_sFilename )
105*b1cdbd2cSJim Jagielski {
106*b1cdbd2cSJim Jagielski 	bool ret = LoadFile(i_sFilename);
107*b1cdbd2cSJim Jagielski 	if (ret)
108*b1cdbd2cSJim Jagielski 		Parse();
109*b1cdbd2cSJim Jagielski 	return ret;
110*b1cdbd2cSJim Jagielski }
111*b1cdbd2cSJim Jagielski 
112*b1cdbd2cSJim Jagielski void
Parse_XmlDeclaration()113*b1cdbd2cSJim Jagielski X2CParser::Parse_XmlDeclaration()
114*b1cdbd2cSJim Jagielski {
115*b1cdbd2cSJim Jagielski 	Goto('<');
116*b1cdbd2cSJim Jagielski 	if ( IsText("<?xml") )
117*b1cdbd2cSJim Jagielski 	{
118*b1cdbd2cSJim Jagielski 		Goto_And_Pass('>');
119*b1cdbd2cSJim Jagielski 	}
120*b1cdbd2cSJim Jagielski }
121*b1cdbd2cSJim Jagielski 
122*b1cdbd2cSJim Jagielski void
Parse_Doctype()123*b1cdbd2cSJim Jagielski X2CParser::Parse_Doctype()
124*b1cdbd2cSJim Jagielski {
125*b1cdbd2cSJim Jagielski 	Goto('<');
126*b1cdbd2cSJim Jagielski 	if ( IsText("<!DOCTYPE") )
127*b1cdbd2cSJim Jagielski 		Goto_And_Pass('>');
128*b1cdbd2cSJim Jagielski }
129*b1cdbd2cSJim Jagielski 
130*b1cdbd2cSJim Jagielski void
Parse_Sequence(DynamicList<XmlElement> & o_rElements,const Simstr & i_sElementName)131*b1cdbd2cSJim Jagielski X2CParser::Parse_Sequence( DynamicList<XmlElement> & o_rElements,
132*b1cdbd2cSJim Jagielski 						   const Simstr &			 i_sElementName  )
133*b1cdbd2cSJim Jagielski {
134*b1cdbd2cSJim Jagielski 	CheckAndPassBeginTag(i_sElementName.str());
135*b1cdbd2cSJim Jagielski 
136*b1cdbd2cSJim Jagielski 	unsigned int i_max = o_rElements.size();
137*b1cdbd2cSJim Jagielski 	for (unsigned i = 0; i < i_max; ++i)
138*b1cdbd2cSJim Jagielski 	{
139*b1cdbd2cSJim Jagielski 		o_rElements[i]->Parse(*this);
140*b1cdbd2cSJim Jagielski 	}  // end for
141*b1cdbd2cSJim Jagielski 
142*b1cdbd2cSJim Jagielski 	CheckAndPassEndTag(i_sElementName.str());
143*b1cdbd2cSJim Jagielski }
144*b1cdbd2cSJim Jagielski 
145*b1cdbd2cSJim Jagielski void
Parse_FreeChoice(DynamicList<XmlElement> & o_rElements)146*b1cdbd2cSJim Jagielski X2CParser::Parse_FreeChoice( DynamicList<XmlElement> & o_rElements )
147*b1cdbd2cSJim Jagielski {
148*b1cdbd2cSJim Jagielski 	unsigned     	nSize = o_rElements.size();
149*b1cdbd2cSJim Jagielski 
150*b1cdbd2cSJim Jagielski 	for ( bool bBreak = false;  !bBreak; )
151*b1cdbd2cSJim Jagielski 	{
152*b1cdbd2cSJim Jagielski 		bBreak = true;
153*b1cdbd2cSJim Jagielski 		for ( unsigned i = 0; i < nSize; ++i )
154*b1cdbd2cSJim Jagielski 		{
155*b1cdbd2cSJim Jagielski 			Goto('<');
156*b1cdbd2cSJim Jagielski 			if ( IsBeginTag(o_rElements[i]->Name().str()) )
157*b1cdbd2cSJim Jagielski 			{
158*b1cdbd2cSJim Jagielski 				o_rElements[i]->Parse(*this);
159*b1cdbd2cSJim Jagielski 				bBreak = false;
160*b1cdbd2cSJim Jagielski 				break;
161*b1cdbd2cSJim Jagielski 			}
162*b1cdbd2cSJim Jagielski 		}  	// end for i
163*b1cdbd2cSJim Jagielski 	} 	// end for !bBreak
164*b1cdbd2cSJim Jagielski }
165*b1cdbd2cSJim Jagielski 
166*b1cdbd2cSJim Jagielski void
Parse_List(ListElement & o_rListElem)167*b1cdbd2cSJim Jagielski X2CParser::Parse_List( ListElement &  o_rListElem )
168*b1cdbd2cSJim Jagielski {
169*b1cdbd2cSJim Jagielski 
170*b1cdbd2cSJim Jagielski 	for ( Goto('<'); IsBeginTag(o_rListElem.Name().str()); Goto('<') )
171*b1cdbd2cSJim Jagielski 	{
172*b1cdbd2cSJim Jagielski 		XmlElement * pNew = o_rListElem.Create_and_Add_NewElement();
173*b1cdbd2cSJim Jagielski 		pNew->Parse(*this);
174*b1cdbd2cSJim Jagielski 	}
175*b1cdbd2cSJim Jagielski }
176*b1cdbd2cSJim Jagielski 
177*b1cdbd2cSJim Jagielski void
Parse_Text(Simstr & o_sText,const Simstr & i_sElementName,bool i_bReverseName)178*b1cdbd2cSJim Jagielski X2CParser::Parse_Text( Simstr &			o_sText,
179*b1cdbd2cSJim Jagielski 					   const Simstr &	i_sElementName,
180*b1cdbd2cSJim Jagielski 					   bool				i_bReverseName )
181*b1cdbd2cSJim Jagielski {
182*b1cdbd2cSJim Jagielski 
183*b1cdbd2cSJim Jagielski 	if ( ! CheckAndPassBeginTag(i_sElementName.str()) )
184*b1cdbd2cSJim Jagielski         return;
185*b1cdbd2cSJim Jagielski 
186*b1cdbd2cSJim Jagielski 	// Add new Element
187*b1cdbd2cSJim Jagielski 	GetTextTill( o_sText, '<', i_bReverseName );
188*b1cdbd2cSJim Jagielski 	o_sText.remove_trailing_blanks();
189*b1cdbd2cSJim Jagielski 
190*b1cdbd2cSJim Jagielski 	CheckAndPassEndTag(i_sElementName.str());
191*b1cdbd2cSJim Jagielski }
192*b1cdbd2cSJim Jagielski 
193*b1cdbd2cSJim Jagielski void
Parse_MultipleText(List<Simstr> & o_rTexts,const Simstr & i_sElementName,bool i_bReverseName)194*b1cdbd2cSJim Jagielski X2CParser::Parse_MultipleText( List<Simstr> &	o_rTexts,
195*b1cdbd2cSJim Jagielski 							   const Simstr &	i_sElementName,
196*b1cdbd2cSJim Jagielski 							   bool				i_bReverseName )
197*b1cdbd2cSJim Jagielski {
198*b1cdbd2cSJim Jagielski 	for ( Goto('<'); IsBeginTag(i_sElementName.str()); Goto('<') )
199*b1cdbd2cSJim Jagielski 	{
200*b1cdbd2cSJim Jagielski         Simstr sNew;
201*b1cdbd2cSJim Jagielski 		Parse_Text(sNew, i_sElementName, i_bReverseName);
202*b1cdbd2cSJim Jagielski         if (sNew.l() > 0)
203*b1cdbd2cSJim Jagielski     		o_rTexts.push_back(sNew);
204*b1cdbd2cSJim Jagielski 	}
205*b1cdbd2cSJim Jagielski }
206*b1cdbd2cSJim Jagielski 
207*b1cdbd2cSJim Jagielski void
Parse_SglAttr(Simstr & o_sAttrValue,const Simstr & i_sElementName,const Simstr & i_sAttrName)208*b1cdbd2cSJim Jagielski X2CParser::Parse_SglAttr( Simstr &			o_sAttrValue,
209*b1cdbd2cSJim Jagielski 						  const Simstr &	i_sElementName,
210*b1cdbd2cSJim Jagielski 						  const Simstr &	i_sAttrName )
211*b1cdbd2cSJim Jagielski {
212*b1cdbd2cSJim Jagielski 	Goto('<');
213*b1cdbd2cSJim Jagielski 	if ( !IsBeginTag(i_sElementName.str()) )
214*b1cdbd2cSJim Jagielski 		SyntaxError("unexpected element");
215*b1cdbd2cSJim Jagielski 	Move( i_sElementName.l() + 1 );
216*b1cdbd2cSJim Jagielski 
217*b1cdbd2cSJim Jagielski 	Pass_White();
218*b1cdbd2cSJim Jagielski 	if (*text == '>')
219*b1cdbd2cSJim Jagielski 		SyntaxError("no attribute found, where one was expected");
220*b1cdbd2cSJim Jagielski 	Simstr sAttrName;
221*b1cdbd2cSJim Jagielski 	Get_Attribute(o_sAttrValue, sAttrName);
222*b1cdbd2cSJim Jagielski 	if (sAttrName != i_sAttrName)
223*b1cdbd2cSJim Jagielski 		SyntaxError("unknown attribute found");
224*b1cdbd2cSJim Jagielski 	Pass_White();
225*b1cdbd2cSJim Jagielski 	if (strncmp(text,"/>",2) != 0)
226*b1cdbd2cSJim Jagielski 		SyntaxError("missing \"/>\" at end of empty element");
227*b1cdbd2cSJim Jagielski 	Move(2);
228*b1cdbd2cSJim Jagielski }
229*b1cdbd2cSJim Jagielski 
230*b1cdbd2cSJim Jagielski void
Parse_MultipleAttr(List<Simstr> & o_rAttrValues,const Simstr & i_sElementName,const List<Simstr> & i_rAttrNames)231*b1cdbd2cSJim Jagielski X2CParser::Parse_MultipleAttr( List<Simstr> &		o_rAttrValues,
232*b1cdbd2cSJim Jagielski 							   const Simstr &		i_sElementName,
233*b1cdbd2cSJim Jagielski 							   const List<Simstr> &	i_rAttrNames )
234*b1cdbd2cSJim Jagielski {
235*b1cdbd2cSJim Jagielski 	Goto('<');
236*b1cdbd2cSJim Jagielski 	if ( !IsBeginTag(i_sElementName.str()) )
237*b1cdbd2cSJim Jagielski 		SyntaxError("unexpected element");
238*b1cdbd2cSJim Jagielski 	Move( i_sElementName.l() + 1 );
239*b1cdbd2cSJim Jagielski 	Simstr sAttrName;
240*b1cdbd2cSJim Jagielski 	Simstr sAttrValue;
241*b1cdbd2cSJim Jagielski 	unsigned nSize = i_rAttrNames.size();
242*b1cdbd2cSJim Jagielski 	unsigned i;
243*b1cdbd2cSJim Jagielski 
244*b1cdbd2cSJim Jagielski 	for ( Pass_White(); *text != '/'; Pass_White() )
245*b1cdbd2cSJim Jagielski 	{
246*b1cdbd2cSJim Jagielski 
247*b1cdbd2cSJim Jagielski 		Get_Attribute(sAttrValue, sAttrName);
248*b1cdbd2cSJim Jagielski 
249*b1cdbd2cSJim Jagielski 		for ( i = 0; i < nSize; ++i )
250*b1cdbd2cSJim Jagielski 		{
251*b1cdbd2cSJim Jagielski 			if ( i_rAttrNames[i] == sAttrName )
252*b1cdbd2cSJim Jagielski             {
253*b1cdbd2cSJim Jagielski 				o_rAttrValues[i] = sAttrValue;
254*b1cdbd2cSJim Jagielski 				break;
255*b1cdbd2cSJim Jagielski 			}
256*b1cdbd2cSJim Jagielski 		}
257*b1cdbd2cSJim Jagielski 		if (i == nSize)
258*b1cdbd2cSJim Jagielski 			SyntaxError("unknown attribute found");
259*b1cdbd2cSJim Jagielski 	}
260*b1cdbd2cSJim Jagielski 	Move(2);
261*b1cdbd2cSJim Jagielski }
262*b1cdbd2cSJim Jagielski 
263*b1cdbd2cSJim Jagielski 
264*b1cdbd2cSJim Jagielski void
Get_Attribute(Simstr & o_rAttrValue,Simstr & o_rAttrName)265*b1cdbd2cSJim Jagielski X2CParser::Get_Attribute( Simstr & o_rAttrValue,
266*b1cdbd2cSJim Jagielski 						  Simstr & o_rAttrName )
267*b1cdbd2cSJim Jagielski {
268*b1cdbd2cSJim Jagielski 	GetTextTill( o_rAttrName, '=');
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski 	while (*(++text) != '"')
271*b1cdbd2cSJim Jagielski 	{
272*b1cdbd2cSJim Jagielski 		if (*text == '\0')
273*b1cdbd2cSJim Jagielski 			SyntaxError("unexpected end of file");
274*b1cdbd2cSJim Jagielski 	}
275*b1cdbd2cSJim Jagielski 
276*b1cdbd2cSJim Jagielski 	++text;
277*b1cdbd2cSJim Jagielski 	GetTextTill( o_rAttrValue, '"');
278*b1cdbd2cSJim Jagielski 	++text;
279*b1cdbd2cSJim Jagielski }
280*b1cdbd2cSJim Jagielski 
281*b1cdbd2cSJim Jagielski bool
IsText(const char * i_sComparedText)282*b1cdbd2cSJim Jagielski X2CParser::IsText( const char *	i_sComparedText )
283*b1cdbd2cSJim Jagielski {
284*b1cdbd2cSJim Jagielski 	return strnicmp( text, i_sComparedText, strlen(i_sComparedText) ) == 0;
285*b1cdbd2cSJim Jagielski }
286*b1cdbd2cSJim Jagielski 
287*b1cdbd2cSJim Jagielski bool
IsBeginTag(const char * i_sTagName)288*b1cdbd2cSJim Jagielski X2CParser::IsBeginTag( const char *	i_sTagName )
289*b1cdbd2cSJim Jagielski {
290*b1cdbd2cSJim Jagielski 	return strnicmp( text+1, i_sTagName, strlen(i_sTagName) ) == 0
291*b1cdbd2cSJim Jagielski 		   && *text == '<';
292*b1cdbd2cSJim Jagielski }
293*b1cdbd2cSJim Jagielski 
294*b1cdbd2cSJim Jagielski bool
IsEndTag(const char * i_sTagName)295*b1cdbd2cSJim Jagielski X2CParser::IsEndTag( const char * i_sTagName )
296*b1cdbd2cSJim Jagielski {
297*b1cdbd2cSJim Jagielski 	return strnicmp( text+2, i_sTagName, strlen(i_sTagName) ) == 0
298*b1cdbd2cSJim Jagielski 		   && strnicmp( text, "</", 2 ) == 0;
299*b1cdbd2cSJim Jagielski }
300*b1cdbd2cSJim Jagielski 
301*b1cdbd2cSJim Jagielski void
Goto(char i_cNext)302*b1cdbd2cSJim Jagielski X2CParser::Goto( char i_cNext )
303*b1cdbd2cSJim Jagielski {
304*b1cdbd2cSJim Jagielski 	while (*text != i_cNext)
305*b1cdbd2cSJim Jagielski 	{
306*b1cdbd2cSJim Jagielski 		TestCurChar();
307*b1cdbd2cSJim Jagielski 		++text;
308*b1cdbd2cSJim Jagielski 	}
309*b1cdbd2cSJim Jagielski }
310*b1cdbd2cSJim Jagielski 
311*b1cdbd2cSJim Jagielski void
Goto_And_Pass(char i_cNext)312*b1cdbd2cSJim Jagielski X2CParser::Goto_And_Pass( char i_cNext )
313*b1cdbd2cSJim Jagielski {
314*b1cdbd2cSJim Jagielski 	Goto(i_cNext);
315*b1cdbd2cSJim Jagielski 	++text;
316*b1cdbd2cSJim Jagielski }
317*b1cdbd2cSJim Jagielski 
318*b1cdbd2cSJim Jagielski void
Move(int i_nForward)319*b1cdbd2cSJim Jagielski X2CParser::Move( int i_nForward )
320*b1cdbd2cSJim Jagielski {
321*b1cdbd2cSJim Jagielski 	text += i_nForward;
322*b1cdbd2cSJim Jagielski }
323*b1cdbd2cSJim Jagielski 
324*b1cdbd2cSJim Jagielski void
Pass_White()325*b1cdbd2cSJim Jagielski X2CParser::Pass_White()
326*b1cdbd2cSJim Jagielski {
327*b1cdbd2cSJim Jagielski 	while (*text <= 32)
328*b1cdbd2cSJim Jagielski 	{
329*b1cdbd2cSJim Jagielski 		TestCurChar();
330*b1cdbd2cSJim Jagielski 		++text;
331*b1cdbd2cSJim Jagielski 	}
332*b1cdbd2cSJim Jagielski }
333*b1cdbd2cSJim Jagielski 
334*b1cdbd2cSJim Jagielski void
GetTextTill(Simstr & o_rText,char i_cEnd,bool i_bReverseName)335*b1cdbd2cSJim Jagielski X2CParser::GetTextTill( Simstr & o_rText,
336*b1cdbd2cSJim Jagielski 						char     i_cEnd,
337*b1cdbd2cSJim Jagielski 						bool     i_bReverseName )
338*b1cdbd2cSJim Jagielski {
339*b1cdbd2cSJim Jagielski 	char * pResult = &sWord[0];
340*b1cdbd2cSJim Jagielski 	char * pSet;
341*b1cdbd2cSJim Jagielski 
342*b1cdbd2cSJim Jagielski 	for ( pSet = pResult;
343*b1cdbd2cSJim Jagielski 		  *text != i_cEnd;
344*b1cdbd2cSJim Jagielski 		  ++text )
345*b1cdbd2cSJim Jagielski 	{
346*b1cdbd2cSJim Jagielski 		TestCurChar();
347*b1cdbd2cSJim Jagielski 		*pSet++ = *text;
348*b1cdbd2cSJim Jagielski 	}
349*b1cdbd2cSJim Jagielski 
350*b1cdbd2cSJim Jagielski 	while ( *pResult < 33 && *pResult > 0 )
351*b1cdbd2cSJim Jagielski 		++pResult;
352*b1cdbd2cSJim Jagielski 	while ( pSet > pResult ? *(pSet-1) < 33 : false )
353*b1cdbd2cSJim Jagielski 		pSet--;
354*b1cdbd2cSJim Jagielski 	*pSet = '\0';
355*b1cdbd2cSJim Jagielski 
356*b1cdbd2cSJim Jagielski 
357*b1cdbd2cSJim Jagielski 	if (i_bReverseName)
358*b1cdbd2cSJim Jagielski 	{
359*b1cdbd2cSJim Jagielski         const unsigned int nMaxLen = 1000;
360*b1cdbd2cSJim Jagielski         if (strlen(pResult) < nMaxLen)
361*b1cdbd2cSJim Jagielski         {
362*b1cdbd2cSJim Jagielski     		char * sBreak = strrchr(pResult,'.');
363*b1cdbd2cSJim Jagielski 	    	if (sBreak != 0)
364*b1cdbd2cSJim Jagielski 		    {
365*b1cdbd2cSJim Jagielski 			    static char sScope[nMaxLen+10];
366*b1cdbd2cSJim Jagielski     			static char sName[nMaxLen+10];
367*b1cdbd2cSJim Jagielski 
368*b1cdbd2cSJim Jagielski         		unsigned nScopeLen = sBreak - pResult;
369*b1cdbd2cSJim Jagielski 		    	strncpy ( sScope, pResult, nScopeLen ); // STRNCPY SAFE HERE
370*b1cdbd2cSJim Jagielski 			    sScope[nScopeLen] = '\0';
371*b1cdbd2cSJim Jagielski 			    strcpy( sName, sBreak + 1 );            // STRCPY SAFE HERE
372*b1cdbd2cSJim Jagielski 			    strcat( sName, " in " );                // STRCAT SAFE HERE
373*b1cdbd2cSJim Jagielski 			    strcat( sName, sScope );                // STRCAT SAFE HERE
374*b1cdbd2cSJim Jagielski 
375*b1cdbd2cSJim Jagielski 			    o_rText = sName;
376*b1cdbd2cSJim Jagielski 			    return;
377*b1cdbd2cSJim Jagielski 		    }
378*b1cdbd2cSJim Jagielski         }
379*b1cdbd2cSJim Jagielski 	}   // endif (i_bReverseName)
380*b1cdbd2cSJim Jagielski 
381*b1cdbd2cSJim Jagielski 	o_rText = &sWord[0];
382*b1cdbd2cSJim Jagielski }
383*b1cdbd2cSJim Jagielski 
384*b1cdbd2cSJim Jagielski bool
CheckAndPassBeginTag(const char * i_sElementName)385*b1cdbd2cSJim Jagielski X2CParser::CheckAndPassBeginTag( const char * i_sElementName )
386*b1cdbd2cSJim Jagielski {
387*b1cdbd2cSJim Jagielski     bool ret = true;
388*b1cdbd2cSJim Jagielski 	Goto('<');
389*b1cdbd2cSJim Jagielski 	if ( ! IsBeginTag(i_sElementName) )
390*b1cdbd2cSJim Jagielski 		SyntaxError( "unexpected element");
391*b1cdbd2cSJim Jagielski     if (IsAbsoluteEmpty())
392*b1cdbd2cSJim Jagielski         ret = false;
393*b1cdbd2cSJim Jagielski 	Goto_And_Pass('>');
394*b1cdbd2cSJim Jagielski     if (ret)
395*b1cdbd2cSJim Jagielski 	    Pass_White();
396*b1cdbd2cSJim Jagielski     return ret;
397*b1cdbd2cSJim Jagielski }
398*b1cdbd2cSJim Jagielski 
399*b1cdbd2cSJim Jagielski void
CheckAndPassEndTag(const char * i_sElementName)400*b1cdbd2cSJim Jagielski X2CParser::CheckAndPassEndTag( const char *	i_sElementName )
401*b1cdbd2cSJim Jagielski {
402*b1cdbd2cSJim Jagielski 	Pass_White();
403*b1cdbd2cSJim Jagielski 	if ( !IsEndTag(i_sElementName) )
404*b1cdbd2cSJim Jagielski 		SyntaxError("missing or not matching end tag");
405*b1cdbd2cSJim Jagielski 	Goto_And_Pass('>');
406*b1cdbd2cSJim Jagielski }
407*b1cdbd2cSJim Jagielski 
408*b1cdbd2cSJim Jagielski bool
IsAbsoluteEmpty() const409*b1cdbd2cSJim Jagielski X2CParser::IsAbsoluteEmpty() const
410*b1cdbd2cSJim Jagielski {
411*b1cdbd2cSJim Jagielski 	const char * pEnd = strchr(text+1, '>');
412*b1cdbd2cSJim Jagielski     if (pEnd != 0)
413*b1cdbd2cSJim Jagielski     {
414*b1cdbd2cSJim Jagielski         if ( *(pEnd-1) == '/' )
415*b1cdbd2cSJim Jagielski         {
416*b1cdbd2cSJim Jagielski             const char * pAttr = strchr(text+1, '"');
417*b1cdbd2cSJim Jagielski          	if (pAttr == 0)
418*b1cdbd2cSJim Jagielski                 return true;
419*b1cdbd2cSJim Jagielski             else if ( (pAttr-text) > (pEnd-text) )
420*b1cdbd2cSJim Jagielski                 return true;
421*b1cdbd2cSJim Jagielski         }
422*b1cdbd2cSJim Jagielski     }
423*b1cdbd2cSJim Jagielski     return false;
424*b1cdbd2cSJim Jagielski }
425*b1cdbd2cSJim Jagielski 
426*b1cdbd2cSJim Jagielski void
SyntaxError(const char * i_sText)427*b1cdbd2cSJim Jagielski X2CParser::SyntaxError( const char * i_sText )
428*b1cdbd2cSJim Jagielski {
429*b1cdbd2cSJim Jagielski 	std::cerr
430*b1cdbd2cSJim Jagielski          << "Syntax error "
431*b1cdbd2cSJim Jagielski 		 << i_sText
432*b1cdbd2cSJim Jagielski 		 << " in file: "
433*b1cdbd2cSJim Jagielski 		 << sFileName.str()
434*b1cdbd2cSJim Jagielski 		 << " in line "
435*b1cdbd2cSJim Jagielski 		 << nFileLine
436*b1cdbd2cSJim Jagielski 		 << "."
437*b1cdbd2cSJim Jagielski 		 << std::endl;
438*b1cdbd2cSJim Jagielski 
439*b1cdbd2cSJim Jagielski 	 exit(3);
440*b1cdbd2cSJim Jagielski }
441*b1cdbd2cSJim Jagielski 
442*b1cdbd2cSJim Jagielski void
TestCurChar()443*b1cdbd2cSJim Jagielski X2CParser::TestCurChar()
444*b1cdbd2cSJim Jagielski {
445*b1cdbd2cSJim Jagielski //	if (*text == '\0')
446*b1cdbd2cSJim Jagielski //		SyntaxError("unexpected end of file");
447*b1cdbd2cSJim Jagielski //	else
448*b1cdbd2cSJim Jagielski 
449*b1cdbd2cSJim Jagielski 	if (*text == '\n')
450*b1cdbd2cSJim Jagielski 		nFileLine++;
451*b1cdbd2cSJim Jagielski }
452*b1cdbd2cSJim Jagielski 
453*b1cdbd2cSJim Jagielski 
454