/* * Copyright (C) 2012 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * Import and export general routing data using a XML file. */ #include "OverrideLog.h" #include "RouteDataSet.h" #include "libxml/xmlmemory.h" #include <errno.h> #include <sys/stat.h> extern char bcm_nfc_location[]; /******************************************************************************* ** ** Function: AidBuffer ** ** Description: Parse a string of hex numbers. Store result in an array of ** bytes. ** aid: string of hex numbers. ** ** Returns: None. ** *******************************************************************************/ AidBuffer::AidBuffer (std::string& aid) : mBuffer (NULL), mBufferLen (0) { unsigned int num = 0; const char delimiter = ':'; std::string::size_type pos1 = 0; std::string::size_type pos2 = aid.find_first_of (delimiter); //parse the AID string; each hex number is separated by a colon; mBuffer = new UINT8 [aid.length()]; while (true) { num = 0; if (pos2 == std::string::npos) { sscanf (aid.substr(pos1).c_str(), "%x", &num); mBuffer [mBufferLen] = (UINT8) num; mBufferLen++; break; } else { sscanf (aid.substr(pos1, pos2-pos1+1).c_str(), "%x", &num); mBuffer [mBufferLen] = (UINT8) num; mBufferLen++; pos1 = pos2 + 1; pos2 = aid.find_first_of (delimiter, pos1); } } } /******************************************************************************* ** ** Function: ~AidBuffer ** ** Description: Release all resources. ** ** Returns: None. ** *******************************************************************************/ AidBuffer::~AidBuffer () { delete [] mBuffer; } /*******************************************************************************/ /*******************************************************************************/ const char* RouteDataSet::sConfigFile = "/param/route.xml"; /******************************************************************************* ** ** Function: ~RouteDataSet ** ** Description: Release all resources. ** ** Returns: None. ** *******************************************************************************/ RouteDataSet::~RouteDataSet () { deleteDatabase (); } /******************************************************************************* ** ** Function: initialize ** ** Description: Initialize resources. ** ** Returns: True if ok. ** *******************************************************************************/ bool RouteDataSet::initialize () { static const char fn [] = "RouteDataSet::initialize"; ALOGD ("%s: enter", fn); //check that the libxml2 version in use is compatible //with the version the software has been compiled with LIBXML_TEST_VERSION ALOGD ("%s: exit; return=true", fn); return true; } /******************************************************************************* ** ** Function: deleteDatabase ** ** Description: Delete all routes stored in all databases. ** ** Returns: None. ** *******************************************************************************/ void RouteDataSet::deleteDatabase () { static const char fn [] = "RouteDataSet::deleteDatabase"; ALOGD ("%s: default db size=%u; sec elem db size=%u", fn, mDefaultRouteDatabase.size(), mSecElemRouteDatabase.size()); Database::iterator it; for (it = mDefaultRouteDatabase.begin(); it != mDefaultRouteDatabase.end(); it++) delete (*it); mDefaultRouteDatabase.clear (); for (it = mSecElemRouteDatabase.begin(); it != mSecElemRouteDatabase.end(); it++) delete (*it); mSecElemRouteDatabase.clear (); } /******************************************************************************* ** ** Function: import ** ** Description: Import data from an XML file. Fill the databases. ** ** Returns: True if ok. ** *******************************************************************************/ bool RouteDataSet::import () { static const char fn [] = "RouteDataSet::import"; ALOGD ("%s: enter", fn); bool retval = false; xmlDocPtr doc; xmlNodePtr node1; std::string strFilename(bcm_nfc_location); strFilename += sConfigFile; deleteDatabase (); doc = xmlParseFile (strFilename.c_str()); if (doc == NULL) { ALOGD ("%s: fail parse", fn); goto TheEnd; } node1 = xmlDocGetRootElement (doc); if (node1 == NULL) { ALOGE ("%s: fail root element", fn); goto TheEnd; } ALOGD ("%s: root=%s", fn, node1->name); node1 = node1->xmlChildrenNode; while (node1) //loop through all elements in <Routes ... { if (xmlStrcmp(node1->name, (const xmlChar*) "Route")==0) { xmlChar* value = xmlGetProp (node1, (const xmlChar*) "Type"); if (value && (xmlStrcmp (value, (const xmlChar*) "SecElemSelectedRoutes") == 0)) { ALOGD ("%s: found SecElemSelectedRoutes", fn); xmlNodePtr node2 = node1->xmlChildrenNode; while (node2) //loop all elements in <Route Type="SecElemSelectedRoutes" ... { if (xmlStrcmp(node2->name, (const xmlChar*) "Proto")==0) importProtocolRoute (node2, mSecElemRouteDatabase); else if (xmlStrcmp(node2->name, (const xmlChar*) "Tech")==0) importTechnologyRoute (node2, mSecElemRouteDatabase); node2 = node2->next; } //loop all elements in <Route Type="SecElemSelectedRoutes" ... } else if (value && (xmlStrcmp (value, (const xmlChar*) "DefaultRoutes") == 0)) { ALOGD ("%s: found DefaultRoutes", fn); xmlNodePtr node2 = node1->xmlChildrenNode; while (node2) //loop all elements in <Route Type="DefaultRoutes" ... { if (xmlStrcmp(node2->name, (const xmlChar*) "Proto")==0) importProtocolRoute (node2, mDefaultRouteDatabase); else if (xmlStrcmp(node2->name, (const xmlChar*) "Tech")==0) importTechnologyRoute (node2, mDefaultRouteDatabase); node2 = node2->next; } //loop all elements in <Route Type="DefaultRoutes" ... } if (value) xmlFree (value); } //check <Route ... node1 = node1->next; } //loop through all elements in <Routes ... retval = true; TheEnd: xmlFreeDoc (doc); xmlCleanupParser (); ALOGD ("%s: exit; return=%u", fn, retval); return retval; } /******************************************************************************* ** ** Function: saveToFile ** ** Description: Save XML data from a string into a file. ** routesXml: XML that represents routes. ** ** Returns: True if ok. ** *******************************************************************************/ bool RouteDataSet::saveToFile (const char* routesXml) { static const char fn [] = "RouteDataSet::saveToFile"; FILE* fh = NULL; size_t actualWritten = 0; bool retval = false; std::string filename (bcm_nfc_location); filename.append (sConfigFile); fh = fopen (filename.c_str (), "w"); if (fh == NULL) { ALOGE ("%s: fail to open file", fn); return false; } actualWritten = fwrite (routesXml, sizeof(char), strlen(routesXml), fh); retval = actualWritten == strlen(routesXml); fclose (fh); ALOGD ("%s: wrote %u bytes", fn, actualWritten); if (retval == false) ALOGE ("%s: error during write", fn); //set file permission to //owner read, write; group read; other read chmod (filename.c_str (), S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH); return retval; } /******************************************************************************* ** ** Function: loadFromFile ** ** Description: Load XML data from file into a string. ** routesXml: string to receive XML data. ** ** Returns: True if ok. ** *******************************************************************************/ bool RouteDataSet::loadFromFile (std::string& routesXml) { static const char fn [] = "RouteDataSet::loadFromFile"; FILE* fh = NULL; size_t actual = 0; char buffer [1024]; std::string filename (bcm_nfc_location); filename.append (sConfigFile); fh = fopen (filename.c_str (), "r"); if (fh == NULL) { ALOGD ("%s: fail to open file", fn); return false; } while (true) { actual = fread (buffer, sizeof(char), sizeof(buffer), fh); if (actual == 0) break; routesXml.append (buffer, actual); } fclose (fh); ALOGD ("%s: read %u bytes", fn, routesXml.length()); return true; } /******************************************************************************* ** ** Function: importProtocolRoute ** ** Description: Parse data for protocol routes. ** element: XML node for one protocol route. ** database: store data in this database. ** ** Returns: None. ** *******************************************************************************/ void RouteDataSet::importProtocolRoute (xmlNodePtr& element, Database& database) { static const char fn [] = "RouteDataSet::importProtocolRoute"; const xmlChar* id = (const xmlChar*) "Id"; const xmlChar* secElem = (const xmlChar*) "SecElem"; const xmlChar* trueString = (const xmlChar*) "true"; const xmlChar* switchOn = (const xmlChar*) "SwitchOn"; const xmlChar* switchOff = (const xmlChar*) "SwitchOff"; const xmlChar* batteryOff = (const xmlChar*) "BatteryOff"; RouteDataForProtocol* data = new RouteDataForProtocol; xmlChar* value = NULL; ALOGD_IF (sDebug, "%s: element=%s", fn, element->name); value = xmlGetProp (element, id); if (value) { if (xmlStrcmp (value, (const xmlChar*) "T1T") == 0) data->mProtocol = NFA_PROTOCOL_MASK_T1T; else if (xmlStrcmp (value, (const xmlChar*) "T2T") == 0) data->mProtocol = NFA_PROTOCOL_MASK_T2T; else if (xmlStrcmp (value, (const xmlChar*) "T3T") == 0) data->mProtocol = NFA_PROTOCOL_MASK_T3T; else if (xmlStrcmp (value, (const xmlChar*) "IsoDep") == 0) data->mProtocol = NFA_PROTOCOL_MASK_ISO_DEP; xmlFree (value); ALOGD_IF (sDebug, "%s: %s=0x%X", fn, id, data->mProtocol); } value = xmlGetProp (element, secElem); if (value) { data->mNfaEeHandle = strtol ((char*) value, NULL, 16); xmlFree (value); data->mNfaEeHandle = data->mNfaEeHandle | NFA_HANDLE_GROUP_EE; ALOGD_IF (sDebug, "%s: %s=0x%X", fn, secElem, data->mNfaEeHandle); } value = xmlGetProp (element, switchOn); if (value) { data->mSwitchOn = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } value = xmlGetProp (element, switchOff); if (value) { data->mSwitchOff = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } value = xmlGetProp (element, batteryOff); if (value) { data->mBatteryOff = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } database.push_back (data); } /******************************************************************************* ** ** Function: importTechnologyRoute ** ** Description: Parse data for technology routes. ** element: XML node for one technology route. ** database: store data in this database. ** ** Returns: None. ** *******************************************************************************/ void RouteDataSet::importTechnologyRoute (xmlNodePtr& element, Database& database) { static const char fn [] = "RouteDataSet::importTechnologyRoute"; const xmlChar* id = (const xmlChar*) "Id"; const xmlChar* secElem = (const xmlChar*) "SecElem"; const xmlChar* trueString = (const xmlChar*) "true"; const xmlChar* switchOn = (const xmlChar*) "SwitchOn"; const xmlChar* switchOff = (const xmlChar*) "SwitchOff"; const xmlChar* batteryOff = (const xmlChar*) "BatteryOff"; RouteDataForTechnology* data = new RouteDataForTechnology; xmlChar* value = NULL; ALOGD_IF (sDebug, "%s: element=%s", fn, element->name); value = xmlGetProp (element, id); if (value) { if (xmlStrcmp (value, (const xmlChar*) "NfcA") == 0) data->mTechnology = NFA_TECHNOLOGY_MASK_A; else if (xmlStrcmp (value, (const xmlChar*) "NfcB") == 0) data->mTechnology = NFA_TECHNOLOGY_MASK_B; else if (xmlStrcmp (value, (const xmlChar*) "NfcF") == 0) data->mTechnology = NFA_TECHNOLOGY_MASK_F; xmlFree (value); ALOGD_IF (sDebug, "%s: %s=0x%X", fn, id, data->mTechnology); } value = xmlGetProp (element, secElem); if (value) { data->mNfaEeHandle = strtol ((char*) value, NULL, 16); xmlFree (value); data->mNfaEeHandle = data->mNfaEeHandle | NFA_HANDLE_GROUP_EE; ALOGD_IF (sDebug, "%s: %s=0x%X", fn, secElem, data->mNfaEeHandle); } value = xmlGetProp (element, switchOn); if (value) { data->mSwitchOn = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } value = xmlGetProp (element, switchOff); if (value) { data->mSwitchOff = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } value = xmlGetProp (element, batteryOff); if (value) { data->mBatteryOff = (xmlStrcmp (value, trueString) == 0); xmlFree (value); } database.push_back (data); } /******************************************************************************* ** ** Function: deleteFile ** ** Description: Delete route data XML file. ** ** Returns: True if ok. ** *******************************************************************************/ bool RouteDataSet::deleteFile () { static const char fn [] = "RouteDataSet::deleteFile"; std::string filename (bcm_nfc_location); filename.append (sConfigFile); int stat = remove (filename.c_str()); ALOGD ("%s: exit %u", fn, stat==0); return stat == 0; } /******************************************************************************* ** ** Function: getDatabase ** ** Description: Obtain a database of routing data. ** selection: which database. ** ** Returns: Pointer to database. ** *******************************************************************************/ RouteDataSet::Database* RouteDataSet::getDatabase (DatabaseSelection selection) { switch (selection) { case DefaultRouteDatabase: return &mDefaultRouteDatabase; case SecElemRouteDatabase: return &mSecElemRouteDatabase; } return NULL; } /******************************************************************************* ** ** Function: printDiagnostic ** ** Description: Print some diagnostic output. ** ** Returns: None. ** *******************************************************************************/ void RouteDataSet::printDiagnostic () { static const char fn [] = "RouteDataSet::printDiagnostic"; Database* db = getDatabase (DefaultRouteDatabase); ALOGD ("%s: default route database", fn); for (Database::iterator iter = db->begin(); iter != db->end(); iter++) { RouteData* routeData = *iter; switch (routeData->mRouteType) { case RouteData::ProtocolRoute: { RouteDataForProtocol* proto = (RouteDataForProtocol*) routeData; ALOGD ("%s: ee h=0x%X; protocol=0x%X", fn, proto->mNfaEeHandle, proto->mProtocol); } break; // TODO: RouteData::TechnologyRoute isn't handled --- bug? } } ALOGD ("%s: sec elem route database", fn); db = getDatabase (SecElemRouteDatabase); for (Database::iterator iter2 = db->begin(); iter2 != db->end(); iter2++) { RouteData* routeData = *iter2; switch (routeData->mRouteType) { case RouteData::ProtocolRoute: { RouteDataForProtocol* proto = (RouteDataForProtocol*) routeData; ALOGD ("%s: ee h=0x%X; protocol=0x%X", fn, proto->mNfaEeHandle, proto->mProtocol); } break; case RouteData::TechnologyRoute: { RouteDataForTechnology* tech = (RouteDataForTechnology*) routeData; ALOGD ("%s: ee h=0x%X; technology=0x%X", fn, tech->mNfaEeHandle, tech->mTechnology); } break; } } }