|  | // Copyright 2016 The SwiftShader Authors. All Rights Reserved. | 
|  | // | 
|  | // 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. | 
|  |  | 
|  | #include "Configurator.hpp" | 
|  |  | 
|  | #include <fstream> | 
|  | #include <iostream> | 
|  |  | 
|  | using namespace std; | 
|  |  | 
|  | #include <ctype.h> | 
|  | #include <stdarg.h> | 
|  | #include <stdio.h> | 
|  |  | 
|  | #if defined(__unix__) | 
|  | #	include <unistd.h> | 
|  | #endif | 
|  |  | 
|  | namespace sw { | 
|  |  | 
|  | Configurator::Configurator(string iniPath) | 
|  | { | 
|  | path = iniPath; | 
|  |  | 
|  | readFile(); | 
|  | } | 
|  |  | 
|  | Configurator::~Configurator() | 
|  | { | 
|  | } | 
|  |  | 
|  | bool Configurator::readFile() | 
|  | { | 
|  | #if defined(__unix__) | 
|  | if(access(path.c_str(), R_OK) != 0) | 
|  | { | 
|  | return false; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | fstream file(path.c_str(), ios::in); | 
|  | if(file.fail()) return false; | 
|  |  | 
|  | string line; | 
|  | string keyName; | 
|  |  | 
|  | while(getline(file, line)) | 
|  | { | 
|  | if(line.length()) | 
|  | { | 
|  | if(line[line.length() - 1] == '\r') | 
|  | { | 
|  | line = line.substr(0, line.length() - 1); | 
|  | } | 
|  |  | 
|  | if(!isprint(line[0])) | 
|  | { | 
|  | //	printf("Failing on char %d\n", line[0]); | 
|  | file.close(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | string::size_type pLeft = line.find_first_of(";#[="); | 
|  |  | 
|  | if(pLeft != string::npos) | 
|  | { | 
|  | switch(line[pLeft]) | 
|  | { | 
|  | case '[': | 
|  | { | 
|  | string::size_type pRight = line.find_last_of("]"); | 
|  |  | 
|  | if(pRight != string::npos && pRight > pLeft) | 
|  | { | 
|  | keyName = line.substr(pLeft + 1, pRight - pLeft - 1); | 
|  | addKeyName(keyName); | 
|  | } | 
|  | } | 
|  | break; | 
|  | case '=': | 
|  | { | 
|  | string valueName = line.substr(0, pLeft); | 
|  | string value = line.substr(pLeft + 1); | 
|  | addValue(keyName, valueName, value); | 
|  | } | 
|  | break; | 
|  | case ';': | 
|  | case '#': | 
|  | // Ignore comments | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | file.close(); | 
|  |  | 
|  | if(names.size()) | 
|  | { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void Configurator::writeFile(std::string title) | 
|  | { | 
|  | #if defined(__unix__) | 
|  | if(access(path.c_str(), W_OK) != 0) | 
|  | { | 
|  | return; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | fstream file(path.c_str(), ios::out); | 
|  | if(file.fail()) return; | 
|  |  | 
|  | file << "; " << title << endl | 
|  | << endl; | 
|  |  | 
|  | for(unsigned int keyID = 0; keyID < sections.size(); keyID++) | 
|  | { | 
|  | file << "[" << names[keyID] << "]" << endl; | 
|  |  | 
|  | for(unsigned int valueID = 0; valueID < sections[keyID].names.size(); valueID++) | 
|  | { | 
|  | file << sections[keyID].names[valueID] << "=" << sections[keyID].values[valueID] << endl; | 
|  | } | 
|  |  | 
|  | file << endl; | 
|  | } | 
|  |  | 
|  | file.close(); | 
|  | } | 
|  |  | 
|  | int Configurator::findKey(string keyName) const | 
|  | { | 
|  | for(unsigned int keyID = 0; keyID < names.size(); keyID++) | 
|  | { | 
|  | if(names[keyID] == keyName) | 
|  | { | 
|  | return keyID; | 
|  | } | 
|  | } | 
|  |  | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | int Configurator::findValue(unsigned int keyID, string valueName) const | 
|  | { | 
|  | if(!sections.size() || keyID >= sections.size()) | 
|  | { | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | for(unsigned int valueID = 0; valueID < sections[keyID].names.size(); ++valueID) | 
|  | { | 
|  | if(sections[keyID].names[valueID] == valueName) | 
|  | { | 
|  | return valueID; | 
|  | } | 
|  | } | 
|  |  | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | unsigned int Configurator::addKeyName(string keyName) | 
|  | { | 
|  | names.resize(names.size() + 1, keyName); | 
|  | sections.resize(sections.size() + 1); | 
|  | return (unsigned int)names.size() - 1; | 
|  | } | 
|  |  | 
|  | void Configurator::addValue(string const keyName, string const valueName, string const value) | 
|  | { | 
|  | int keyID = findKey(keyName); | 
|  |  | 
|  | if(keyID == -1) | 
|  | { | 
|  | keyID = addKeyName(keyName); | 
|  | } | 
|  |  | 
|  | int valueID = findValue(keyID, valueName); | 
|  |  | 
|  | if(valueID == -1) | 
|  | { | 
|  | sections[keyID].names.resize(sections[keyID].names.size() + 1, valueName); | 
|  | sections[keyID].values.resize(sections[keyID].values.size() + 1, value); | 
|  | } | 
|  | else | 
|  | { | 
|  | sections[keyID].values[valueID] = value; | 
|  | } | 
|  | } | 
|  |  | 
|  | string Configurator::getValue(string keyName, string valueName, string defaultValue) const | 
|  | { | 
|  | int keyID = findKey(keyName); | 
|  | if(keyID == -1) return defaultValue; | 
|  | int valueID = findValue((unsigned int)keyID, valueName); | 
|  | if(valueID == -1) return defaultValue; | 
|  |  | 
|  | return sections[keyID].values[valueID]; | 
|  | } | 
|  |  | 
|  | int Configurator::getInteger(string keyName, string valueName, int defaultValue) const | 
|  | { | 
|  | char svalue[256]; | 
|  |  | 
|  | sprintf(svalue, "%d", defaultValue); | 
|  |  | 
|  | return atoi(getValue(keyName, valueName, svalue).c_str()); | 
|  | } | 
|  |  | 
|  | bool Configurator::getBoolean(string keyName, string valueName, bool defaultValue) const | 
|  | { | 
|  | return getInteger(keyName, valueName, (int)defaultValue) != 0; | 
|  | } | 
|  |  | 
|  | double Configurator::getFloat(string keyName, string valueName, double defaultValue) const | 
|  | { | 
|  | char svalue[256]; | 
|  |  | 
|  | sprintf(svalue, "%f", defaultValue); | 
|  |  | 
|  | return atof(getValue(keyName, valueName, svalue).c_str()); | 
|  | } | 
|  |  | 
|  | unsigned int Configurator::getFormatted(string keyName, string valueName, char *format, | 
|  | void *v1, void *v2, void *v3, void *v4, | 
|  | void *v5, void *v6, void *v7, void *v8, | 
|  | void *v9, void *v10, void *v11, void *v12, | 
|  | void *v13, void *v14, void *v15, void *v16) | 
|  | { | 
|  | string value = getValue(keyName, valueName); | 
|  |  | 
|  | if(!value.length()) return false; | 
|  |  | 
|  | unsigned int nVals = sscanf(value.c_str(), format, | 
|  | v1, v2, v3, v4, v5, v6, v7, v8, | 
|  | v9, v10, v11, v12, v13, v14, v15, v16); | 
|  |  | 
|  | return nVals; | 
|  | } | 
|  |  | 
|  | }  // namespace sw |