--HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%401442
		
			
				
	
	
		
			310 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
|  * vim: set ts=4 :
 | |
|  * =============================================================================
 | |
|  * SourceMod
 | |
|  * Copyright (C) 2004-2007 AlliedModders LLC.  All rights reserved.
 | |
|  * =============================================================================
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it under
 | |
|  * the terms of the GNU General Public License, version 3.0, as published by the
 | |
|  * Free Software Foundation.
 | |
|  * 
 | |
|  * This program is distributed in the hope that it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 | |
|  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
 | |
|  * details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License along with
 | |
|  * this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * As a special exception, AlliedModders LLC gives you permission to link the
 | |
|  * code of this program (as well as its derivative works) to "Half-Life 2," the
 | |
|  * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
 | |
|  * by the Valve Corporation.  You must obey the GNU General Public License in
 | |
|  * all respects for all other code used.  Additionally, AlliedModders LLC grants
 | |
|  * this exception to all derivative works.  AlliedModders LLC defines further
 | |
|  * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
 | |
|  * or <http://www.sourcemod.net/license.php>.
 | |
|  *
 | |
|  * Version: $Id$
 | |
|  */
 | |
| 
 | |
| #include <stdarg.h>
 | |
| #include <string.h>
 | |
| #include "PluginInfoDatabase.h"
 | |
| #include "PluginSys.h"
 | |
| 
 | |
| void PluginSettings::Init()
 | |
| {
 | |
| 	name = -1;
 | |
| 	pause_val = false;
 | |
| 	type_val = PluginType_MapUpdated;
 | |
| 	optarray = -1;
 | |
| 	opts_num = 0;
 | |
| 	opts_size = 0;
 | |
| 	blockload_val = false;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * :TODO: write the logger, make these errors log instead of being static
 | |
|  * NOTE: once we do that, we will have to change some code to ignore sections
 | |
|  */
 | |
| 
 | |
| CPluginInfoDatabase::CPluginInfoDatabase()
 | |
| {
 | |
| 	m_strtab = NULL;
 | |
| 	m_infodb_count = 0;
 | |
| 	m_infodb_size = 0;
 | |
| 	m_infodb = -1;
 | |
| }
 | |
| 
 | |
| CPluginInfoDatabase::~CPluginInfoDatabase()
 | |
| {
 | |
| 	delete m_strtab;
 | |
| }
 | |
| 
 | |
| void CPluginInfoDatabase::ReadSMC_ParseStart()
 | |
| {
 | |
| 	/* Create or reset our string table */
 | |
| 	if (m_strtab)
 | |
| 	{
 | |
| 		m_strtab->Reset();
 | |
| 	} else {
 | |
| 		m_strtab = new BaseStringTable(1024);
 | |
| 	}
 | |
| 
 | |
| 	/* Set our internal states to the beginning */
 | |
| 	in_plugins = false;
 | |
| 	cur_plugin = -1;
 | |
| 	in_options = false;
 | |
| 	m_infodb_size = 0;
 | |
| 	m_infodb_count = 0;
 | |
| 	m_infodb = -1;
 | |
| }
 | |
| 
 | |
| SMCParseResult CPluginInfoDatabase::MakeError(const char *fmt, ...)
 | |
| {
 | |
| 	char buffer[512];
 | |
| 	va_list ap;
 | |
| 
 | |
| 	va_start(ap, fmt);
 | |
| 	vsnprintf(buffer, sizeof(buffer), fmt, ap);
 | |
| 	va_end(ap);
 | |
| 
 | |
| 	m_errmsg = m_strtab->AddString(buffer);
 | |
| 
 | |
| 	return SMCParse_HaltFail;
 | |
| }
 | |
| 
 | |
| unsigned int CPluginInfoDatabase::GetSettingsNum()
 | |
| {
 | |
| 	return m_infodb_count;
 | |
| }
 | |
| 
 | |
| PluginSettings *CPluginInfoDatabase::GetSettingsIfMatch(unsigned int index, const char *filename)
 | |
| {
 | |
| 	BaseMemTable *memtab = m_strtab->GetMemTable();
 | |
| 	int *table = (int *)memtab->GetAddress(m_infodb);
 | |
| 
 | |
| 	if (!table || index >= m_infodb_count)
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	PluginSettings *plugin = (PluginSettings *)memtab->GetAddress(table[index]);
 | |
| 
 | |
| 	const char *name = m_strtab->GetString(plugin->name);
 | |
| 
 | |
| 	if (!name)
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!g_PluginSys.TestAliasMatch(name, filename))
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return plugin;
 | |
| }
 | |
| 
 | |
| void CPluginInfoDatabase::GetOptionsForPlugin(PluginSettings *settings, unsigned int opt_num, const char **key, const char **val)
 | |
| {
 | |
| 	PluginOpts *table = (PluginOpts *)m_strtab->GetMemTable()->GetAddress(settings->optarray);
 | |
| 	if (!table)
 | |
| 	{
 | |
| 		*key = NULL;
 | |
| 		*val = NULL;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (opt_num >= settings->opts_num)
 | |
| 	{
 | |
| 		*key = NULL;
 | |
| 		*val = NULL;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	*key = m_strtab->GetString(table[opt_num].key);
 | |
| 	*val = m_strtab->GetString(table[opt_num].val);
 | |
| }
 | |
| 
 | |
| SMCParseResult CPluginInfoDatabase::ReadSMC_KeyValue(const char *key, 
 | |
| 													 const char *value, 
 | |
| 													 bool key_quotes, 
 | |
| 													 bool value_quotes)
 | |
| {
 | |
| 	if (cur_plugin != -1)
 | |
| 	{
 | |
| 		PluginSettings *plugin = (PluginSettings *)m_strtab->GetMemTable()->GetAddress(cur_plugin);
 | |
| 		if (!in_options)
 | |
| 		{
 | |
| 			if (strcmp(key, "pause") == 0)
 | |
| 			{
 | |
| 				if (strcasecmp(value, "yes") == 0)
 | |
| 				{
 | |
| 					plugin->pause_val = true;
 | |
| 				} else {
 | |
| 					plugin->pause_val = false;
 | |
| 				}
 | |
| 			} else if (strcmp(key, "lifetime") == 0) {
 | |
| 				if (strcasecmp(value, "private") == 0)
 | |
| 				{
 | |
| 					plugin->type_val = PluginType_Private;
 | |
| 				} else if (strcasecmp(value, "mapsync") == 0) {
 | |
| 					plugin->type_val = PluginType_MapUpdated;
 | |
| 				} else if (strcasecmp(value, "maponly") == 0) {
 | |
| 					plugin->type_val = PluginType_MapOnly;
 | |
| 				} else if (strcasecmp(value, "global") == 0) {
 | |
| 					plugin->type_val = PluginType_Global;
 | |
| 				} else {
 | |
| 					return MakeError("Unknown value for key \"lifetime\": \"%s\"", value);
 | |
| 				}
 | |
| 			} else if (strcmp(key, "blockload") == 0) {
 | |
| 				plugin->blockload_val = true;
 | |
| 			} else {
 | |
| 				return MakeError("Unknown property key: \"%s\"", key);
 | |
| 			}
 | |
| 		} else {
 | |
| 			/* Cache every option, valid or not */
 | |
| 			int keyidx = m_strtab->AddString(key);
 | |
| 			int validx = m_strtab->AddString(value);
 | |
| 			PluginOpts *table;
 | |
| 			BaseMemTable *memtab = m_strtab->GetMemTable();
 | |
| 			plugin = (PluginSettings *)memtab->GetAddress(cur_plugin);
 | |
| 			if (plugin->opts_num + 1 > plugin->opts_size)
 | |
| 			{
 | |
| 				unsigned int oldsize = plugin->opts_size;
 | |
| 				if (oldsize == 0)
 | |
| 				{
 | |
| 					//right now we don't have many
 | |
| 					plugin->opts_size = 2;
 | |
| 				} else {
 | |
| 					plugin->opts_size *= 2;
 | |
| 				}
 | |
| 				int newidx = memtab->CreateMem(plugin->opts_size * sizeof(PluginOpts), (void **)&table);
 | |
| 				/* in case it resized */
 | |
| 				plugin = (PluginSettings *)memtab->GetAddress(cur_plugin);
 | |
| 				if (plugin->optarray != -1)
 | |
| 				{
 | |
| 					void *oldtable = memtab->GetAddress(plugin->optarray);
 | |
| 					memcpy(table, oldtable, oldsize * sizeof(PluginOpts));
 | |
| 				}
 | |
| 				plugin->optarray = newidx;
 | |
| 			} else {
 | |
| 				table = (PluginOpts *)memtab->GetAddress(plugin->optarray);
 | |
| 			}
 | |
| 			PluginOpts *opt = &table[plugin->opts_num++];
 | |
| 			opt->key = keyidx;
 | |
| 			opt->val = validx;
 | |
| 		}
 | |
| 	} else if (in_plugins) {
 | |
| 		return MakeError("Unknown property key: \"%s\"", key);
 | |
| 	} else {
 | |
| 		/* Ignore anything we don't know about! */
 | |
| 	}
 | |
| 
 | |
| 	return SMCParse_Continue;
 | |
| }
 | |
| 
 | |
| SMCParseResult CPluginInfoDatabase::ReadSMC_LeavingSection()
 | |
| {
 | |
| 	if (in_plugins)
 | |
| 	{
 | |
| 		if (cur_plugin != -1)
 | |
| 		{
 | |
| 			if (in_options)
 | |
| 			{
 | |
| 				in_options = false;
 | |
| 			} else {
 | |
| 				/* If the plugin is ending, add it to the table */
 | |
| 				BaseMemTable *memtab = m_strtab->GetMemTable();
 | |
| 				int *table;
 | |
| 				if (m_infodb_count + 1 > m_infodb_size)
 | |
| 				{
 | |
| 					unsigned int oldsize = m_infodb_size;
 | |
| 					if (!m_infodb_size)
 | |
| 					{
 | |
| 						m_infodb_size = 8;
 | |
| 					} else {
 | |
| 						m_infodb_size *= 2;
 | |
| 					}
 | |
| 					int newidx = memtab->CreateMem(m_infodb_size, (void **)&table);
 | |
| 					if (m_infodb != -1)
 | |
| 					{
 | |
| 						void *oldtable = (int *)memtab->GetAddress(m_infodb);
 | |
| 						memcpy(table, oldtable, oldsize * sizeof(int));
 | |
| 					}
 | |
| 					m_infodb = newidx;
 | |
| 				} else {
 | |
| 					table = (int *)memtab->GetAddress(m_infodb);
 | |
| 				}
 | |
| 				/* Assign to table and scrap the current plugin */
 | |
| 				table[m_infodb_count++] = cur_plugin;
 | |
| 				cur_plugin = -1;
 | |
| 			}
 | |
| 		} else {
 | |
| 			in_plugins = false;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return SMCParse_Continue;
 | |
| }
 | |
| 
 | |
| SMCParseResult CPluginInfoDatabase::ReadSMC_NewSection(const char *name, bool opt_quotes)
 | |
| {
 | |
| 	if (!in_plugins)
 | |
| 	{
 | |
| 		/* If we're not in the main Plugins section, and we don't get it for the name, error out */
 | |
| 		if (strcmp(name, "Plugins") != 0)
 | |
| 		{
 | |
| 			return MakeError("Unknown root section: \"%s\"", name);
 | |
| 		} else {
 | |
| 			/* Otherwise set our states */
 | |
| 			in_plugins = true;
 | |
| 			cur_plugin = -1;
 | |
| 			in_options = false;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (cur_plugin == -1)
 | |
| 		{
 | |
| 			/* If we get a plugin node and we don't have a current plugin, create a new one */
 | |
| 			PluginSettings *plugin;
 | |
| 			int i_name = m_strtab->AddString(name);
 | |
| 			cur_plugin = m_strtab->GetMemTable()->CreateMem(sizeof(PluginSettings), (void **)&plugin);
 | |
| 			plugin->Init();
 | |
| 			plugin->name = i_name;
 | |
| 			in_options = false;
 | |
| 		} else {
 | |
| 			if (!in_options && strcmp(name, "Options") == 0)
 | |
| 			{
 | |
| 				in_options = true;
 | |
| 			} else {
 | |
| 				return MakeError("Unknown plugin sub-section: \"%s\"", name);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return SMCParse_Continue;
 | |
| }
 |