NSIS/Source/lineparse.cpp

239 lines
5.1 KiB
C++

/*
* lineparse.cpp
*
* This file is a part of NSIS.
*
* Copyright (C) 1999-2009 Nullsoft and Contributors
*
* Licensed under the zlib/libpng license (the "License");
* you may not use this file except in compliance with the License.
*
* Licence details can be found in the file COPYING.
*
* This software is provided 'as-is', without any express or implied
* warranty.
*
* Unicode support by Jim Park -- 08/09/2007
*/
#include "lineparse.h"
#include "Platform.h"
#include "tchar.h"
#include <cstdlib>
#include <cstring>
#include "tstring.h"
LineParser::LineParser(bool bCommentBlock)
{
m_incommentblock=bCommentBlock;
m_incomment=false;
m_nt=m_eat=0;
m_tokens=0;
}
LineParser::~LineParser()
{
freetokens();
}
bool LineParser::inComment()
{
return m_incomment;
}
bool LineParser::inCommentBlock()
{
return m_incommentblock;
}
int LineParser::parse(TCHAR *line, int ignore_escaping/*=0*/) // returns -1 on error
{
freetokens();
bool bPrevCB=m_incommentblock;
int n=doline(line, ignore_escaping);
if (n) return n;
if (m_nt)
{
m_incommentblock=bPrevCB;
m_tokens=(TCHAR**)malloc(sizeof(TCHAR*)*m_nt);
n=doline(line, ignore_escaping);
if (n)
{
freetokens();
return -1;
}
}
return 0;
}
int LineParser::getnumtokens()
{
return m_nt-m_eat;
}
void LineParser::eattoken()
{
m_eat++;
}
double LineParser::gettoken_float(int token, int *success/*=0*/)
{
token+=m_eat;
if (token < 0 || token >= m_nt)
{
if (success) *success=0;
return 0.0;
}
if (success)
{
TCHAR *t=m_tokens[token];
*success=*t?1:0;
while (*t)
{
if ((*t < _T('0') || *t > _T('9'))&&*t != _T('.')) *success=0;
t++;
}
}
return _tstof(m_tokens[token]);
}
int LineParser::gettoken_int(int token, int *success/*=0*/)
{
token+=m_eat;
if (token < 0 || token >= m_nt || !m_tokens[token][0])
{
if (success) *success=0;
return 0;
}
TCHAR *tmp;
int l;
if (m_tokens[token][0] == _T('-')) l=_tcstol(m_tokens[token],&tmp,0);
else l=(int)_tcstoul(m_tokens[token],&tmp,0);
if (success) *success=! (int)(*tmp);
return l;
}
TCHAR* LineParser::gettoken_str(int token)
{
token+=m_eat;
if (token < 0 || token >= m_nt) return (TCHAR*)_T("");
return m_tokens[token];
}
int LineParser::gettoken_enum(int token, const TCHAR *strlist) // null seperated list
{
int x=0;
TCHAR *tt=gettoken_str(token);
if (tt && *tt) while (*strlist)
{
if (!_tcsicmp(tt,strlist)) return x;
strlist+=_tcsclen(strlist)+1;
x++;
}
return -1;
}
void LineParser::freetokens()
{
if (m_tokens)
{
int x;
for (x = 0; x < m_nt; x ++)
free(m_tokens[x]);
free(m_tokens);
}
m_tokens=0;
m_nt=0;
}
int LineParser::doline(TCHAR *line, int ignore_escaping/*=0*/)
{
m_nt=0;
m_incomment = false;
while (*line == _T(' ') || *line == _T('\t')) line++;
while (*line)
{
if ( m_incommentblock )
{
while ( *line )
{
if ( *line == _T('*') && *(line+1) == _T('/') )
{
m_incommentblock=false; // Found end of comment block
line+=2;
while (*line == _T(' ') || *line == _T('\t')) line++;
break;
}
else line++;
}
}
else {
int lstate=0; // 1=", 2=`, 4='
if (*line == _T(';') || *line == _T('#'))
{
m_incomment = true;
break;
}
if (*line == _T('/') && *(line+1) == _T('*'))
{
m_incommentblock = true;
line+=2;
}
else {
if (*line == _T('\"')) lstate=1;
else if (*line == _T('\'')) lstate=2;
else if (*line == _T('`')) lstate=4;
if (lstate) line++;
int nc=0;
TCHAR *p = line;
while (*line)
{
if (line[0] == _T('$') && line[1] == _T('\\')) {
switch (line[2]) {
case _T('"'):
case _T('\''):
case _T('`'):
nc += ignore_escaping ? 3 : 1;
line += 3;
continue;
}
}
if (lstate==1 && *line ==_T('\"')) break;
if (lstate==2 && *line ==_T('\'')) break;
if (lstate==4 && *line ==_T('`')) break;
if (!lstate && (*line == _T(' ') || *line == _T('\t'))) break;
#ifdef NSIS_FIX_COMMENT_HANDLING
if (!lstate && (*line == _T(';') || *line == _T('#') || (*line == _T('/') && *(line+1) == _T('*')))) break;
#endif
line++;
nc++;
}
if (m_tokens)
{
int i;
m_tokens[m_nt]=(TCHAR*)malloc((nc+1)*sizeof(TCHAR));
for (i = 0; p < line; i++, p++) {
if (!ignore_escaping && p[0] == _T('$') && p[1] == _T('\\')) {
switch (p[2]) {
case _T('"'):
case _T('\''):
case _T('`'):
p += 2;
}
}
m_tokens[m_nt][i] = *p;
}
m_tokens[m_nt][nc]=0;
}
m_nt++;
if (lstate)
{
if (*line) line++;
else return -2;
}
while (*line == _T(' ') || *line == _T('\t')) line++;
}
}
}
return 0;
}