Files
SingularityViewer/indra/llxml/llxmlparser.cpp
Lirusaito 2653567d82 Sync with alchemy
Moves some constants out of llavatarconstants.h
8ebf8f4608bd: Change various const constants to constexpr
e07d5d43ba30: CID-143595
30b6935fc66d: CID-143595
acc96f9051cb: Fix a memory leak in viewer side baking initial wearable setup
Sync llmodel.*

Changesets assimilated:
f8f7706c2902: CID-143554 - fix out of bounds access
223eb65adce4: CID-143554 - Chase
2ceb49aaa133: CID-42838, CID-42930, CID-42933, CID-42938, CID-42940, CID-42945, CID-42948, CID-56111, CID-83907
d220005d9f23: Missing null check before deref
31dbb0f3b6ee: CID-42571 CID-42576 CID-42578
49caf082e65c: change unordered_map to flat_map
Doesn't cause as many problems as a hashmap when it comes to assumptions in the
LLUI system.
f93f5e881484: "update" linux cef
downgrade to fix javascript problems
cba818dd9269: Various null checks and etc.
1b4c6bc483bb: CID-42847, CID-42854, CID-42886, CID-42921, CID-42922, CID-42923, CID-42924, CID-42925,
CID-42927, CID-42928, CID-83871, CID-83876, CID-83878, CID-83880, CID-83900, CID-143573
0fe90cd9ec24: Various file size related things
a79f6f653dca: CID-42918 - Initialize member pointers in LLFloaterGodTools
0b70d600d978: Tweak LLFloaterBuyLand initializations
e8b173ffe813: CID-42854 - Additional fix to LLDrawInfo
b5d745cf3fde: Fix signage
4f2e2f384781: Initialize and cleanup various class member variables.
CID-42899, CID-42900, CID-42902, CID-42903, CID-42904, CID-42905, CID-42909, CID-42910,
CID-42911, CID-42912, CID-42913, CID-42967, CID-83853, CID-83898, CID-83890, CID-143584
9851a3e39b4c: Fix platform specific include directories
5c074e84f1be: Initialize and clenaup various more class member variables.
CID-42885, CID-42853, CID-42894, CID-42895, CID-42896, CID-83908, CID-143574, CID-143575,
CID-143576, CID-143576, CID-143578
ac262854ac92: Brace sub-object in initialization to make our intentions clear to clang
358da477d4c1: More double brace init
c3850119314a: Initialize various member pointers in panels
CID-83902, CID-83903, CID-83905, CID-83909, CID-83911, CID-83912, CID-143572
2016-03-30 21:06:47 -04:00

423 lines
9.5 KiB
C++

/**
* @file llxmlparser.cpp
* @brief LLXmlParser implementation
*
* $LicenseInfo:firstyear=2002&license=viewergpl$
*
* Copyright (c) 2002-2009, Linden Research, Inc.
*
* Second Life Viewer Source Code
* The source code in this file ("Source Code") is provided by Linden Lab
* to you under the terms of the GNU General Public License, version 2.0
* ("GPL"), unless you have obtained a separate licensing agreement
* ("Other License"), formally executed by you and Linden Lab. Terms of
* the GPL can be found in doc/GPL-license.txt in this distribution, or
* online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
*
* There are special exceptions to the terms and conditions of the GPL as
* it is applied to this Source Code. View the full text of the exception
* in the file doc/FLOSS-exception.txt in this software distribution, or
* online at
* http://secondlifegrid.net/programs/open_source/licensing/flossexception
*
* By copying, modifying or distributing this software, you acknowledge
* that you have read and understood your obligations described above,
* and agree to abide by those obligations.
*
* ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
* WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
* COMPLETENESS OR PERFORMANCE.
* $/LicenseInfo$
*/
// llxmlparser.cpp
//
// copyright 2002, linden research inc
#include "linden_common.h"
#include "llxmlparser.h"
#include "llerror.h"
LLXmlParser::LLXmlParser()
:
mParser( NULL ),
mDepth( 0 )
{
mAuxErrorString = "no error";
// Override the document's declared encoding.
mParser = XML_ParserCreate(NULL);
XML_SetUserData(mParser, this);
XML_SetElementHandler( mParser, startElementHandler, endElementHandler);
XML_SetCharacterDataHandler( mParser, characterDataHandler);
XML_SetProcessingInstructionHandler( mParser, processingInstructionHandler);
XML_SetCommentHandler( mParser, commentHandler);
XML_SetCdataSectionHandler( mParser, startCdataSectionHandler, endCdataSectionHandler);
// This sets the default handler but does not inhibit expansion of internal entities.
// The entity reference will not be passed to the default handler.
XML_SetDefaultHandlerExpand( mParser, defaultDataHandler);
XML_SetUnparsedEntityDeclHandler( mParser, unparsedEntityDeclHandler);
}
LLXmlParser::~LLXmlParser()
{
XML_ParserFree( mParser );
}
BOOL LLXmlParser::parseFile(const std::string &path)
{
llassert( !mDepth );
BOOL success = TRUE;
LLFILE* file = LLFile::fopen(path, "rb"); /* Flawfinder: ignore */
if( !file )
{
mAuxErrorString = llformat( "Couldn't open file %s", path.c_str());
success = FALSE;
}
else
{
S32 bytes_read = 0;
fseek(file, 0L, SEEK_END);
size_t buffer_size = ftell(file);
fseek(file, 0L, SEEK_SET);
void* buffer = XML_GetBuffer(mParser, buffer_size);
if( !buffer )
{
mAuxErrorString = llformat( "Unable to allocate XML buffer while reading file %s", path.c_str() );
success = FALSE;
goto exit_label;
}
bytes_read = (S32)fread(buffer, 1, buffer_size, file);
if( bytes_read <= 0 )
{
mAuxErrorString = llformat( "Error while reading file %s", path.c_str() );
success = FALSE;
goto exit_label;
}
if( !XML_ParseBuffer(mParser, bytes_read, TRUE ) )
{
mAuxErrorString = llformat( "Error while parsing file %s", path.c_str() );
success = FALSE;
}
exit_label:
fclose( file );
}
if( success )
{
llassert( !mDepth );
}
mDepth = 0;
if( !success )
{
LL_WARNS() << mAuxErrorString << LL_ENDL;
}
return success;
}
// Parses some input. Returns 0 if a fatal error is detected.
// The last call must have isFinal true;
// len may be zero for this call (or any other).
S32 LLXmlParser::parse( const char* buf, int len, int isFinal )
{
return XML_Parse(mParser, buf, len, isFinal);
}
const char* LLXmlParser::getErrorString()
{
const char* error_string = XML_ErrorString(XML_GetErrorCode( mParser ));
if( !error_string )
{
error_string = mAuxErrorString.c_str();
}
return error_string;
}
S32 LLXmlParser::getCurrentLineNumber()
{
return XML_GetCurrentLineNumber( mParser );
}
S32 LLXmlParser::getCurrentColumnNumber()
{
return XML_GetCurrentColumnNumber(mParser);
}
///////////////////////////////////////////////////////////////////////////////
// Pseudo-private methods. These are only used by internal callbacks.
// static
void LLXmlParser::startElementHandler(
void *userData,
const XML_Char *name,
const XML_Char **atts)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->startElement( name, atts );
self->mDepth++;
}
// static
void LLXmlParser::endElementHandler(
void *userData,
const XML_Char *name)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->mDepth--;
self->endElement( name );
}
// s is not 0 terminated.
// static
void LLXmlParser::characterDataHandler(
void *userData,
const XML_Char *s,
int len)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->characterData( s, len );
}
// target and data are 0 terminated
// static
void LLXmlParser::processingInstructionHandler(
void *userData,
const XML_Char *target,
const XML_Char *data)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->processingInstruction( target, data );
}
// data is 0 terminated
// static
void LLXmlParser::commentHandler(void *userData, const XML_Char *data)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->comment( data );
}
// static
void LLXmlParser::startCdataSectionHandler(void *userData)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->mDepth++;
self->startCdataSection();
}
// static
void LLXmlParser::endCdataSectionHandler(void *userData)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->endCdataSection();
self->mDepth++;
}
// This is called for any characters in the XML document for
// which there is no applicable handler. This includes both
// characters that are part of markup which is of a kind that is
// not reported (comments, markup declarations), or characters
// that are part of a construct which could be reported but
// for which no handler has been supplied. The characters are passed
// exactly as they were in the XML document except that
// they will be encoded in UTF-8. Line boundaries are not normalized.
// Note that a byte order mark character is not passed to the default handler.
// There are no guarantees about how characters are divided between calls
// to the default handler: for example, a comment might be split between
// multiple calls.
// static
void LLXmlParser::defaultDataHandler(
void *userData,
const XML_Char *s,
int len)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->defaultData( s, len );
}
// This is called for a declaration of an unparsed (NDATA)
// entity. The base argument is whatever was set by XML_SetBase.
// The entityName, systemId and notationName arguments will never be null.
// The other arguments may be.
// static
void LLXmlParser::unparsedEntityDeclHandler(
void *userData,
const XML_Char *entityName,
const XML_Char *base,
const XML_Char *systemId,
const XML_Char *publicId,
const XML_Char *notationName)
{
LLXmlParser* self = (LLXmlParser*) userData;
self->unparsedEntityDecl( entityName, base, systemId, publicId, notationName );
}
////////////////////////////////////////////////////////////////////
// Test code.
/*
class LLXmlDOMParser : public LLXmlParser
{
public:
LLXmlDOMParser() {}
virtual ~LLXmlDOMParser() {}
void tabs()
{
for ( int i = 0; i < getDepth(); i++)
{
putchar(' ');
}
}
virtual void startElement(const char *name, const char **atts)
{
tabs();
printf("startElement %s\n", name);
S32 i = 0;
while( atts[i] && atts[i+1] )
{
tabs();
printf( "\t%s=%s\n", atts[i], atts[i+1] );
i += 2;
}
if( atts[i] )
{
tabs();
printf( "\ttrailing attribute: %s\n", atts[i] );
}
}
virtual void endElement(const char *name)
{
tabs();
printf("endElement %s\n", name);
}
virtual void characterData(const char *s, int len)
{
tabs();
char* str = new char[len+1];
strncpy( str, s, len );
str[len] = '\0';
printf("CharacterData %s\n", str);
delete str;
}
virtual void processingInstruction(const char *target, const char *data)
{
tabs();
printf("processingInstruction %s\n", data);
}
virtual void comment(const char *data)
{
tabs();
printf("comment %s\n", data);
}
virtual void startCdataSection()
{
tabs();
printf("startCdataSection\n");
}
virtual void endCdataSection()
{
tabs();
printf("endCdataSection\n");
}
virtual void defaultData(const char *s, int len)
{
tabs();
char* str = new char[len+1];
strncpy( str, s, len );
str[len] = '\0';
printf("defaultData %s\n", str);
delete str;
}
virtual void unparsedEntityDecl(
const char *entityName,
const char *base,
const char *systemId,
const char *publicId,
const char *notationName)
{
tabs();
printf(
"unparsed entity:\n"
"\tentityName %s\n"
"\tbase %s\n"
"\tsystemId %s\n"
"\tpublicId %s\n"
"\tnotationName %s\n",
entityName,
base,
systemId,
publicId,
notationName );
}
};
int main()
{
char buf[1024];
LLFILE* file = LLFile::fopen("test.xml", "rb");
if( !file )
{
return 1;
}
LLXmlDOMParser parser;
int done;
do {
size_t len = fread(buf, 1, sizeof(buf), file);
done = len < sizeof(buf);
if( 0 == parser.parse( buf, len, done) )
{
fprintf(stderr,
"%s at line %d\n",
parser.getErrorString(),
parser.getCurrentLineNumber() );
return 1;
}
} while (!done);
fclose( file );
return 0;
}
*/