2009-09-30 23:10:58 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* HomebrewXML Class
|
|
|
|
* for USB Loader GX
|
|
|
|
***************************************************************************/
|
2010-10-24 19:08:03 +00:00
|
|
|
#include <gctypes.h>
|
2009-09-30 23:10:58 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2010-10-24 19:08:03 +00:00
|
|
|
#include "FileOperations/fileops.h"
|
2016-11-06 15:01:54 +00:00
|
|
|
#include "xml/tinyxml2.h"
|
2011-06-14 17:53:19 +00:00
|
|
|
#include "gecko.h"
|
2009-09-30 23:10:58 +00:00
|
|
|
|
|
|
|
#include "HomebrewXML.h"
|
|
|
|
|
2016-11-06 15:01:54 +00:00
|
|
|
using namespace tinyxml2;
|
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
#define ENTRIE_SIZE 8192
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2010-09-24 00:48:03 +00:00
|
|
|
/* qparam filename Filepath of the XML file */
|
|
|
|
int HomebrewXML::LoadHomebrewXMLData(const char* filename)
|
2010-09-17 16:15:18 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
Name.clear();
|
|
|
|
Coder.clear();
|
|
|
|
Version.clear();
|
|
|
|
ShortDescription.clear();
|
|
|
|
LongDescription.clear();
|
|
|
|
Releasedate.clear();
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2016-11-06 15:01:54 +00:00
|
|
|
XMLDocument xmlDoc;
|
|
|
|
if(xmlDoc.LoadFile(filename) != 0)
|
2011-07-25 22:28:22 +00:00
|
|
|
return false;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2016-11-06 15:01:54 +00:00
|
|
|
XMLElement *appNode = xmlDoc.FirstChildElement("app");
|
2012-02-29 19:52:36 +00:00
|
|
|
if(!appNode)
|
|
|
|
return false;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2016-11-06 15:01:54 +00:00
|
|
|
XMLElement *node = NULL;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("name");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
Name = node->FirstChild()->Value();
|
2010-10-24 19:08:03 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("coder");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
Coder = node->FirstChild()->Value();
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("version");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
Version = node->FirstChild()->Value();
|
2010-09-17 16:15:18 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("short_description");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
ShortDescription = node->FirstChild()->Value();
|
2010-09-17 16:15:18 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("long_description");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
LongDescription = node->FirstChild()->Value();
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
char ReleaseText[200];
|
|
|
|
memset(ReleaseText, 0, sizeof(ReleaseText));
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("release_date");
|
|
|
|
if(node && node->FirstChild() && node->FirstChild()->Value())
|
2011-07-25 22:28:22 +00:00
|
|
|
snprintf(ReleaseText, sizeof(ReleaseText), node->FirstChild()->Value());
|
2010-09-17 16:15:18 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
int len = (strlen(ReleaseText) - 6); //length of the date string without the 200000 at the end
|
|
|
|
if (len == 8)
|
|
|
|
snprintf(ReleaseText, sizeof(ReleaseText), "%c%c/%c%c/%c%c%c%c", ReleaseText[4], ReleaseText[5], ReleaseText[6], ReleaseText[7], ReleaseText[0], ReleaseText[1], ReleaseText[2], ReleaseText[3]);
|
|
|
|
else if (len == 6)
|
|
|
|
snprintf(ReleaseText, sizeof(ReleaseText), "%c%c/%c%c%c%c", ReleaseText[4], ReleaseText[5], ReleaseText[0], ReleaseText[1], ReleaseText[2], ReleaseText[3]);
|
2010-09-17 16:15:18 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
Releasedate = ReleaseText;
|
2009-09-30 23:10:58 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
node = appNode->FirstChildElement("arguments");
|
|
|
|
if(!node)
|
|
|
|
return 1;
|
2011-06-14 17:53:19 +00:00
|
|
|
|
2016-11-06 15:01:54 +00:00
|
|
|
XMLElement *argNode = node->FirstChildElement("arg");
|
2011-06-14 17:53:19 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
while(argNode)
|
|
|
|
{
|
|
|
|
if(argNode->FirstChild() && argNode->FirstChild()->Value())
|
|
|
|
Arguments.push_back(std::string(argNode->FirstChild()->Value()));
|
2010-09-17 16:15:18 +00:00
|
|
|
|
2011-06-25 19:50:57 +00:00
|
|
|
argNode = argNode->NextSiblingElement();
|
2011-07-25 22:28:22 +00:00
|
|
|
}
|
2010-10-24 19:08:03 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
return 1;
|
2009-09-30 23:10:58 +00:00
|
|
|
}
|
2010-09-24 00:48:03 +00:00
|
|
|
|
2014-05-04 14:42:52 +00:00
|
|
|
int HomebrewXML::SaveHomebrewXMLData(const char* filename)
|
|
|
|
{
|
|
|
|
const int max_line_size = 4096;
|
2015-03-08 16:49:07 +00:00
|
|
|
char *line = new (std::nothrow) char[max_line_size];
|
|
|
|
if(!line) return 0;
|
2014-05-04 14:42:52 +00:00
|
|
|
|
|
|
|
FILE *fp = fopen(filename, "wb");
|
|
|
|
if(!fp)
|
|
|
|
{
|
|
|
|
delete [] line;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
snprintf(line, max_line_size,"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>\n"); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size,"<app version=\"1\">\n"); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <name>%s</name>\n", GetName()); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <coder>%s</coder>\n", GetCoder()); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <version>%s</version>\n", GetVersion()); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <release_date>%s</release_date>\n", GetReleasedate()); fputs(line, fp);
|
|
|
|
if (Arguments.size() > 0)
|
|
|
|
{
|
|
|
|
snprintf(line, max_line_size," <arguments>\n"); fputs(line, fp);
|
|
|
|
for(u8 i = 0; i < Arguments.size(); i++)
|
|
|
|
{
|
|
|
|
snprintf(line, max_line_size," <arg>%s</arg>\n", Arguments[i].c_str()); fputs(line, fp);
|
|
|
|
}
|
|
|
|
snprintf(line, max_line_size," </arguments>\n"); fputs(line, fp);
|
|
|
|
}
|
|
|
|
snprintf(line, max_line_size," <ahb_access/>\n"); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <short_description>%s</short_description>\n", GetShortDescription()); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size," <long_description>%s</long_description>\n", GetLongDescription()); fputs(line, fp);
|
|
|
|
snprintf(line, max_line_size,"</app>\n"); fputs(line, fp);
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
delete [] line;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set argument */
|
|
|
|
void HomebrewXML::SetArgument(const char* argument)
|
|
|
|
{
|
|
|
|
// Crop value from argument, if present
|
|
|
|
char argName[strlen(argument)+1];
|
|
|
|
strcpy(argName, argument);
|
|
|
|
char *ptr = strrchr(argName, '=');
|
|
|
|
if(ptr) *(ptr+1) = 0;
|
|
|
|
|
|
|
|
// Check if argument already exists and edit it
|
|
|
|
bool found = false;
|
|
|
|
for(u8 i=0; i < Arguments.size(); i++)
|
|
|
|
{
|
|
|
|
size_t pos = Arguments[i].find(argName);
|
|
|
|
if(pos != std::string::npos)
|
|
|
|
{
|
|
|
|
Arguments[i] = argument;
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if it doesn't exist, add the new argument.
|
|
|
|
if(!found)
|
|
|
|
Arguments.push_back(argument);
|
|
|
|
}
|
|
|
|
|
2010-09-24 00:48:03 +00:00
|
|
|
/* Get name */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetName() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return Name.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Set Name */
|
|
|
|
void HomebrewXML::SetName(char * newName)
|
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
Name = newName;
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Get coder */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetCoder() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return Coder.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Get version */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetVersion() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return Version.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
2014-05-04 14:42:52 +00:00
|
|
|
/* Set version */
|
|
|
|
void HomebrewXML::SetVersion(const char * newVer)
|
|
|
|
{
|
|
|
|
Version = newVer;
|
|
|
|
}
|
|
|
|
|
2010-09-24 00:48:03 +00:00
|
|
|
/* Get releasedate */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetReleasedate() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return Releasedate.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Get shortdescription */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetShortDescription() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return ShortDescription.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Get longdescription */
|
2010-10-24 19:08:03 +00:00
|
|
|
const char * HomebrewXML::GetLongDescription() const
|
2010-09-24 00:48:03 +00:00
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
return LongDescription.c_str();
|
2010-09-24 00:48:03 +00:00
|
|
|
}
|