gtkradiant/tools/quake2/common/scriplib.c
2012-03-17 15:01:54 -05:00

299 lines
6.1 KiB
C

/*
Copyright (C) 1999-2007 id Software, Inc. and contributors.
For a list of contributors, see the accompanying CONTRIBUTORS file.
This file is part of GtkRadiant.
GtkRadiant is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
GtkRadiant 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 GtkRadiant; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
// scriplib.c
#include "cmdlib.h"
#include "inout.h"
#include "scriplib.h"
/*
=============================================================================
PARSING STUFF
=============================================================================
*/
typedef struct
{
char filename[1024];
char *buffer,*script_p,*end_p;
int line;
} script_t;
#define MAX_INCLUDES 8
script_t scriptstack[MAX_INCLUDES];
script_t *script;
int scriptline;
char token[MAXTOKEN];
qboolean endofscript;
qboolean tokenready; // only true if UnGetToken was just called
/*
==============
AddScriptToStack
==============
*/
void AddScriptToStack( char *filename ){
int size;
script++;
if ( script == &scriptstack[MAX_INCLUDES] ) {
Error( "script file exceeded MAX_INCLUDES" );
}
strcpy( script->filename, ExpandPath( filename ) );
size = LoadFile( script->filename, (void **)&script->buffer );
printf( "entering %s\n", script->filename );
script->line = 1;
script->script_p = script->buffer;
script->end_p = script->buffer + size;
}
/*
==============
LoadScriptFile
==============
*/
void LoadScriptFile( char *filename ){
script = scriptstack;
AddScriptToStack( filename );
endofscript = false;
tokenready = false;
}
/*
==============
ParseFromMemory
==============
*/
void ParseFromMemory( char *buffer, int size ){
script = scriptstack;
script++;
if ( script == &scriptstack[MAX_INCLUDES] ) {
Error( "script file exceeded MAX_INCLUDES" );
}
strcpy( script->filename, "memory buffer" );
script->buffer = buffer;
script->line = 1;
script->script_p = script->buffer;
script->end_p = script->buffer + size;
endofscript = false;
tokenready = false;
}
/*
==============
UnGetToken
Signals that the current token was not used, and should be reported
for the next GetToken. Note that
GetToken (true);
UnGetToken ();
GetToken (false);
could cross a line boundary.
==============
*/
void UnGetToken( void ){
tokenready = true;
}
qboolean EndOfScript( qboolean crossline ){
if ( !crossline ) {
Error( "Line %i is incomplete\n",scriptline );
}
if ( !strcmp( script->filename, "memory buffer" ) ) {
endofscript = true;
return false;
}
free( script->buffer );
if ( script == scriptstack + 1 ) {
endofscript = true;
return false;
}
script--;
scriptline = script->line;
printf( "returning to %s\n", script->filename );
return GetToken( crossline );
}
/*
==============
GetToken
==============
*/
qboolean GetToken( qboolean crossline ){
char *token_p;
if ( tokenready ) { // is a token allready waiting?
tokenready = false;
return true;
}
if ( script->script_p >= script->end_p ) {
return EndOfScript( crossline );
}
//
// skip space
//
skipspace:
while ( *script->script_p <= 32 )
{
if ( script->script_p >= script->end_p ) {
return EndOfScript( crossline );
}
if ( *script->script_p++ == '\n' ) {
if ( !crossline ) {
Error( "Line %i is incomplete\n",scriptline );
}
scriptline = script->line++;
}
}
if ( script->script_p >= script->end_p ) {
return EndOfScript( crossline );
}
// ; # // comments
if ( *script->script_p == ';' || *script->script_p == '#'
|| ( script->script_p[0] == '/' && script->script_p[1] == '/' ) ) {
if ( !crossline ) {
Error( "Line %i is incomplete\n",scriptline );
}
while ( *script->script_p++ != '\n' )
if ( script->script_p >= script->end_p ) {
return EndOfScript( crossline );
}
goto skipspace;
}
// /* */ comments
if ( script->script_p[0] == '/' && script->script_p[1] == '*' ) {
if ( !crossline ) {
Error( "Line %i is incomplete\n",scriptline );
}
script->script_p += 2;
while ( script->script_p[0] != '*' && script->script_p[1] != '/' )
{
script->script_p++;
if ( script->script_p >= script->end_p ) {
return EndOfScript( crossline );
}
}
script->script_p += 2;
goto skipspace;
}
//
// copy token
//
token_p = token;
if ( *script->script_p == '"' ) {
// quoted token
script->script_p++;
while ( *script->script_p != '"' )
{
*token_p++ = *script->script_p++;
if ( script->script_p == script->end_p ) {
break;
}
if ( token_p == &token[MAXTOKEN] ) {
Error( "Token too large on line %i\n",scriptline );
}
}
script->script_p++;
}
else{ // regular token
while ( *script->script_p > 32 && *script->script_p != ';' )
{
*token_p++ = *script->script_p++;
if ( script->script_p == script->end_p ) {
break;
}
if ( token_p == &token[MAXTOKEN] ) {
Error( "Token too large on line %i\n",scriptline );
}
}
}
*token_p = 0;
if ( !strcmp( token, "$include" ) ) {
GetToken( false );
AddScriptToStack( token );
return GetToken( crossline );
}
return true;
}
/*
==============
TokenAvailable
Returns true if there is another token on the line
==============
*/
qboolean TokenAvailable( void ){
char *search_p;
search_p = script->script_p;
if ( search_p >= script->end_p ) {
return false;
}
while ( *search_p <= 32 )
{
if ( *search_p == '\n' ) {
return false;
}
search_p++;
if ( search_p == script->end_p ) {
return false;
}
}
if ( *search_p == ';' ) {
return false;
}
return true;
}