///###////////////////////////////////////////////////////////////////////////
//
// Burton Computer Corporation
// http://www.burton-computer.com
// http://www.cooldevtools.com
// $Id: HdlParser.cc 272 2007-01-06 19:37:27Z brian $
//
// Copyright (C) 2007 Burton Computer Corporation
// ALL RIGHTS RESERVED
//
// This program is open source software; you can redistribute it
// and/or modify it under the terms of the Q Public License (QPL)
// version 1.0. Use of this software in whole or in part, including
// linking it (modified or unmodified) into other programs is
// subject to the terms of the QPL.
//
// 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
// Q Public License for more details.
//
// You should have received a copy of the Q Public License
// along with this program; see the file LICENSE.txt. If not, visit
// the Burton Computer Corporation or CoolDevTools web site
// QPL pages at:
//
// http://www.burton-computer.com/qpl.html
// http://www.cooldevtools.com/qpl.html
//
#include <fstream>
#include "IstreamCharReader.h"
#include "StringReader.h"
#include "HdlError.h"
#include "HdlToken.h"
#include "HdlTokenizer.h"
#include "HdlStatement.h"
#include "HdlParser.h"
const Ref<HdlStatement> HdlParser::parseString(const string &src)
{
StringReader reader(src);
m_tokenizer = make_ref(new HdlTokenizer("<string>", &reader));
return parse();
}
const Ref<HdlStatement> HdlParser::parseFile(const File &file)
{
ifstream fin(file.getPath().c_str());
IstreamCharReader reader(&fin);
m_tokenizer = make_ref(new HdlTokenizer(file.getPath(), &reader));
return parse();
}
const Ref<HdlStatement> HdlParser::parse()
{
Ref<HdlStatement> stmt;
Ref<HdlToken> id(skipAndEnsureIdentifier());
if (id->isBegin()) {
stmt = parseBlock();
} else {
stmt = parseStatement(id, false);
}
m_tokenizer.clear();
return stmt;
}
const Ref<HdlStatement> HdlParser::parseBlock()
{
Ref<HdlToken> id(skipAndEnsureIdentifier());
Ref<HdlStatement> stmt(parseStatement(id, true));
for (;;) {
id = skipAndEnsureIdentifier();
if (id->isBegin()) {
Ref<HdlStatement> child(parseBlock());
stmt->addChild(child);
} else if (id->isEnd()) {
skipAndEnsureSemiColon();
return stmt;
} else {
Ref<HdlStatement> child(parseStatement(id, false));
stmt->addChild(child);
}
}
}
void HdlParser::skipAndEnsureSemiColon()
{
if (!m_tokenizer->nextToken()) {
throw HdlError("unexpected eof", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
if (!m_tokenizer->isSemiColon()) {
throw HdlError("expected semi-colon", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
}
const Ref<HdlToken> HdlParser::skipAndEnsureIdentifier()
{
if (!m_tokenizer->nextToken()) {
throw HdlError("unexpected eof", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
if (!m_tokenizer->hasToken() || m_tokenizer->token().isNull() || !m_tokenizer->token()->isIdentifier()) {
throw HdlError("expected identifier", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
return m_tokenizer->takeToken();
}
const Ref<HdlStatement> HdlParser::parseStatement(const Ref<HdlToken> &id,
bool is_block)
{
if (id->isBegin() || id->isEnd()) {
throw HdlError(string("invalid statement name: ") + id->strValue(), m_tokenizer->filename(), m_tokenizer->lineNumber());
}
Ref<HdlStatement> stmt(new HdlStatement(id, is_block));
parseArgumentsForStatement(stmt);
return stmt;
}
void HdlParser::parseArgumentsForStatement(const Ref<HdlStatement> &stmt)
{
for (;;) {
if (!m_tokenizer->nextToken()) {
throw HdlError("unexpected eof", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
if (m_tokenizer->isSemiColon()) {
break;
}
if (!m_tokenizer->hasToken()) {
// this probably can't happen but just being safe
throw HdlError("token expected", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
Ref<HdlToken> id(m_tokenizer->takeToken());
if (id->isBegin() || id->isEnd()) {
throw HdlError("invalid use of begin/end", m_tokenizer->filename(), m_tokenizer->lineNumber());
}
stmt->addArgument(id);
}
}
syntax highlighted by Code2HTML, v. 0.9.1