387 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			387 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * $Id$
 | 
						|
 *
 | 
						|
    Copyright (c) 2006-2020 Chung, Hyung-Hwan. All rights reserved.
 | 
						|
 | 
						|
    Redistribution and use in source and binary forms, with or without
 | 
						|
    modification, are permitted provided that the following conditions
 | 
						|
    are met:
 | 
						|
    1. Redistributions of source code must retain the above copyright
 | 
						|
       notice, this list of conditions and the following disclaimer.
 | 
						|
    2. Redistributions in binary form must reproduce the above copyright
 | 
						|
       notice, this list of conditions and the following disclaimer in the
 | 
						|
       documentation and/or other materials provided with the distribution.
 | 
						|
 | 
						|
    THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
 | 
						|
    IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 | 
						|
    OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 | 
						|
    IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 | 
						|
    INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 | 
						|
    NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | 
						|
    DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | 
						|
    THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | 
						|
    (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 | 
						|
    THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
						|
 */
 | 
						|
 | 
						|
#include <Hawk-Sed.hpp>
 | 
						|
#include "sed-prv.h"
 | 
						|
#include "hawk-prv.h"
 | 
						|
 | 
						|
/////////////////////////////////
 | 
						|
HAWK_BEGIN_NAMESPACE(HAWK)
 | 
						|
/////////////////////////////////
 | 
						|
 | 
						|
struct sed_xtn_t
 | 
						|
{
 | 
						|
	Sed* sed;
 | 
						|
};
 | 
						|
 | 
						|
#if defined(HAWK_HAVE_INLINE)
 | 
						|
static HAWK_INLINE sed_xtn_t* GET_XTN(hawk_sed_t* sed) { return (sed_xtn_t*)((hawk_uint8_t*)hawk_sed_getxtn(sed) - HAWK_SIZEOF(sed_xtn_t)); }
 | 
						|
#else
 | 
						|
#define GET_XTN(sed) ((sed_xtn_t*)((hawk_uint8_t*)hawk_sed_getxtn(sed) - HAWK_SIZEOF(sed_xtn_t)))
 | 
						|
#endif
 | 
						|
 | 
						|
Sed::Sed (Mmgr* mmgr): Mmged(mmgr), sed(HAWK_NULL), dflerrstr(HAWK_NULL)
 | 
						|
{
 | 
						|
	HAWK_MEMSET (&this->errinf, 0, HAWK_SIZEOF(this->errinf));
 | 
						|
	this->errinf.num = HAWK_ENOERR;
 | 
						|
	this->_cmgr = hawk_get_cmgr_by_id(HAWK_CMGR_UTF8);
 | 
						|
}
 | 
						|
 | 
						|
hawk_cmgr_t* Sed::getCmgr () const
 | 
						|
{
 | 
						|
	return this->sed? hawk_sed_getcmgr(this->sed): this->_cmgr;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::setCmgr (hawk_cmgr_t* cmgr)
 | 
						|
{
 | 
						|
	if (this->sed) hawk_sed_setcmgr(this->sed, cmgr);
 | 
						|
	this->_cmgr = cmgr;
 | 
						|
}
 | 
						|
 | 
						|
int Sed::open ()
 | 
						|
{
 | 
						|
	this->sed = hawk_sed_open(this->getMmgr(), HAWK_SIZEOF(sed_xtn_t), this->getCmgr(), &this->errinf);
 | 
						|
	if (HAWK_UNLIKELY(!this->sed)) return -1;
 | 
						|
 | 
						|
	this->sed->_instsize += HAWK_SIZEOF(sed_xtn_t);
 | 
						|
 | 
						|
	sed_xtn_t* xtn = GET_XTN(this->sed);
 | 
						|
	xtn->sed = this;
 | 
						|
 | 
						|
	this->dflerrstr = hawk_sed_geterrstr(this->sed);
 | 
						|
// TODO: revive this too when hawk_sed_seterrstr is revived()
 | 
						|
//	hawk_sed_seterrstr (this->sed, Sed::xerrstr);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::close ()
 | 
						|
{
 | 
						|
	if (this->sed)
 | 
						|
	{
 | 
						|
		hawk_sed_close(this->sed);
 | 
						|
		this->sed = HAWK_NULL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
int Sed::compile (Stream& sstream)
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	this->sstream = &sstream;
 | 
						|
	int n = hawk_sed_comp(this->sed, Sed::sin);
 | 
						|
	if (n <= -1) this->retrieveError();
 | 
						|
	return n;
 | 
						|
}
 | 
						|
 | 
						|
int Sed::execute (Stream& istream, Stream& ostream)
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	this->istream = &istream;
 | 
						|
	this->ostream = &ostream;
 | 
						|
	int n = hawk_sed_exec(this->sed, Sed::xin, Sed::xout);
 | 
						|
	if (n <= -1) this->retrieveError();
 | 
						|
	return n;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::halt ()
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	hawk_sed_halt(this->sed);
 | 
						|
}
 | 
						|
 | 
						|
bool Sed::isHalt () const
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	return hawk_sed_ishalt(this->sed);
 | 
						|
}
 | 
						|
 | 
						|
int Sed::getTrait () const
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	int val;
 | 
						|
	hawk_sed_getopt(this->sed, HAWK_SED_TRAIT, &val);
 | 
						|
	return val;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::setTrait (int trait)
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	hawk_sed_setopt(this->sed, HAWK_SED_TRAIT, &trait);
 | 
						|
}
 | 
						|
 | 
						|
const hawk_ooch_t* Sed::getErrorMessage () const
 | 
						|
{
 | 
						|
	return this->errinf.msg;
 | 
						|
}
 | 
						|
 | 
						|
const hawk_uch_t* Sed::getErrorMessageU () const
 | 
						|
{
 | 
						|
#if defined(HAWK_OOCH_IS_UCH)
 | 
						|
	return this->errinf.msg;
 | 
						|
#else
 | 
						|
	hawk_oow_t wcslen, mbslen;
 | 
						|
	wcslen = HAWK_COUNTOF(this->xerrmsg);
 | 
						|
	hawk_conv_bcstr_to_ucstr_with_cmgr(this->errinf.msg, &mbslen, this->xerrmsg, &wcslen, this->getCmgr(), 1);
 | 
						|
	return this->xerrmsg;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
const hawk_bch_t* Sed::getErrorMessageB () const
 | 
						|
{
 | 
						|
#if defined(HAWK_OOCH_IS_UCH)
 | 
						|
	hawk_oow_t wcslen, mbslen;
 | 
						|
	mbslen = HAWK_COUNTOF(this->xerrmsg);
 | 
						|
	hawk_conv_ucstr_to_bcstr_with_cmgr(this->errinf.msg, &wcslen, this->xerrmsg, &mbslen, this->getCmgr());
 | 
						|
	return this->xerrmsg;
 | 
						|
#else
 | 
						|
	return this->errinf.msg;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
hawk_loc_t Sed::getErrorLocation () const
 | 
						|
{
 | 
						|
	return this->errinf.loc;
 | 
						|
}
 | 
						|
 | 
						|
const hawk_uch_t* Sed::getErrorLocationFileU () const
 | 
						|
{
 | 
						|
#if defined(HAWK_OOCH_IS_UCH)
 | 
						|
	return this->errinf.loc.file;
 | 
						|
#else
 | 
						|
	if (!this->errinf.loc.file) return HAWK_NULL;
 | 
						|
	hawk_oow_t wcslen, mbslen;
 | 
						|
	wcslen = HAWK_COUNTOF(this->xerrlocfile);
 | 
						|
	hawk_conv_bcstr_to_ucstr_with_cmgr(this->errinf.loc.file, &mbslen, this->xerrlocfile, &wcslen, this->getCmgr(), 1);
 | 
						|
	return this->xerrlocfile;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
const hawk_bch_t* Sed::getErrorLocationFileB () const
 | 
						|
{
 | 
						|
#if defined(HAWK_OOCH_IS_UCH)
 | 
						|
	if (!this->errinf.loc.file) return HAWK_NULL;
 | 
						|
	hawk_oow_t wcslen, mbslen;
 | 
						|
	mbslen = HAWK_COUNTOF(this->xerrlocfile);
 | 
						|
	hawk_conv_ucstr_to_bcstr_with_cmgr(this->errinf.loc.file, &wcslen, this->xerrlocfile, &mbslen, this->getCmgr());
 | 
						|
	return this->xerrlocfile;
 | 
						|
#else
 | 
						|
	return this->errinf.loc.file;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
hawk_errnum_t Sed::getErrorNumber () const
 | 
						|
{
 | 
						|
	return this->errinf.num;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::setError (hawk_errnum_t code, const hawk_oocs_t* args, const hawk_loc_t* loc)
 | 
						|
{
 | 
						|
	if (this->sed)
 | 
						|
	{
 | 
						|
		hawk_sed_seterror(this->sed, loc, code, args);
 | 
						|
		this->retrieveError();
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		HAWK_MEMSET(&this->errinf, 0, HAWK_SIZEOF(this->errinf));
 | 
						|
		this->errinf.num = code;
 | 
						|
		if (loc) this->errinf.loc = *loc;
 | 
						|
		hawk_copy_oocstr(this->errinf.msg, HAWK_COUNTOF(this->errinf.msg), hawk_dfl_errstr(code));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Sed::formatError (hawk_errnum_t code, const hawk_loc_t* loc, const hawk_bch_t* fmt, ...)
 | 
						|
{
 | 
						|
	if (this->sed)
 | 
						|
	{
 | 
						|
		va_list ap;
 | 
						|
		va_start(ap, fmt);
 | 
						|
		hawk_sed_seterrbvfmt(this->sed, loc, code, fmt, ap);
 | 
						|
		va_end(ap);
 | 
						|
		this->retrieveError();
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		HAWK_MEMSET(&this->errinf, 0, HAWK_SIZEOF(this->errinf));
 | 
						|
		this->errinf.num = code;
 | 
						|
		if (loc) this->errinf.loc = *loc;
 | 
						|
		hawk_copy_oocstr(this->errinf.msg, HAWK_COUNTOF(this->errinf.msg), hawk_dfl_errstr(code));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Sed::formatError (hawk_errnum_t code, const hawk_loc_t* loc, const hawk_uch_t* fmt, ...)
 | 
						|
{
 | 
						|
	if (this->sed)
 | 
						|
	{
 | 
						|
		va_list ap;
 | 
						|
		va_start(ap, fmt);
 | 
						|
		hawk_sed_seterruvfmt(this->sed, loc, code, fmt, ap);
 | 
						|
		va_end(ap);
 | 
						|
		this->retrieveError();
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		HAWK_MEMSET(&this->errinf, 0, HAWK_SIZEOF(this->errinf));
 | 
						|
		this->errinf.num = code;
 | 
						|
		if (loc) this->errinf.loc = *loc;
 | 
						|
		hawk_copy_oocstr(this->errinf.msg, HAWK_COUNTOF(this->errinf.msg), hawk_dfl_errstr(code));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Sed::clearError ()
 | 
						|
{
 | 
						|
	HAWK_MEMSET(&this->errinf, 0, HAWK_SIZEOF(this->errinf));
 | 
						|
	this->errinf.num = HAWK_ENOERR;
 | 
						|
}
 | 
						|
 | 
						|
void Sed::retrieveError()
 | 
						|
{
 | 
						|
	if (this->sed == HAWK_NULL)
 | 
						|
	{
 | 
						|
		this->clearError();
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		hawk_sed_geterrinf(this->sed, &this->errinf);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
const hawk_ooch_t* Sed::getCompileId () const
 | 
						|
{
 | 
						|
	return hawk_sed_getcompid(this->sed);
 | 
						|
}
 | 
						|
 | 
						|
const hawk_ooch_t* Sed::setCompileId (const hawk_ooch_t* id)
 | 
						|
{
 | 
						|
	return hawk_sed_setcompid(this->sed, id);
 | 
						|
}
 | 
						|
 | 
						|
hawk_oow_t Sed::getConsoleLine ()
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	return hawk_sed_getlinenum(this->sed);
 | 
						|
}
 | 
						|
 | 
						|
void Sed::setConsoleLine (hawk_oow_t num)
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->sed != HAWK_NULL);
 | 
						|
	hawk_sed_setlinenum (this->sed, num);
 | 
						|
}
 | 
						|
 | 
						|
hawk_ooi_t Sed::sin (hawk_sed_t* s, hawk_sed_io_cmd_t cmd, hawk_sed_io_arg_t* arg, hawk_ooch_t* buf, hawk_oow_t len)
 | 
						|
{
 | 
						|
	sed_xtn_t* xtn = GET_XTN(s);
 | 
						|
 | 
						|
	Stream::Data iodata (xtn->sed, Stream::READ, arg);
 | 
						|
 | 
						|
	try
 | 
						|
	{
 | 
						|
		switch (cmd)
 | 
						|
		{
 | 
						|
			case HAWK_SED_IO_OPEN:
 | 
						|
				return xtn->sed->sstream->open(iodata);
 | 
						|
			case HAWK_SED_IO_CLOSE:
 | 
						|
				return xtn->sed->sstream->close(iodata);
 | 
						|
			case HAWK_SED_IO_READ:
 | 
						|
				return xtn->sed->sstream->read(iodata, buf, len);
 | 
						|
			default:
 | 
						|
				return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	catch (...)
 | 
						|
	{
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
hawk_ooi_t Sed::xin (hawk_sed_t* s, hawk_sed_io_cmd_t cmd, hawk_sed_io_arg_t* arg, hawk_ooch_t* buf, hawk_oow_t len)
 | 
						|
{
 | 
						|
	sed_xtn_t* xtn = GET_XTN(s);
 | 
						|
 | 
						|
	Stream::Data iodata (xtn->sed, Stream::READ, arg);
 | 
						|
 | 
						|
	try
 | 
						|
	{
 | 
						|
		switch (cmd)
 | 
						|
		{
 | 
						|
			case HAWK_SED_IO_OPEN:
 | 
						|
				return xtn->sed->istream->open(iodata);
 | 
						|
			case HAWK_SED_IO_CLOSE:
 | 
						|
				return xtn->sed->istream->close(iodata);
 | 
						|
			case HAWK_SED_IO_READ:
 | 
						|
				return xtn->sed->istream->read(iodata, buf, len);
 | 
						|
			default:
 | 
						|
				return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	catch (...)
 | 
						|
	{
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
hawk_ooi_t Sed::xout (hawk_sed_t* s, hawk_sed_io_cmd_t cmd, hawk_sed_io_arg_t* arg, hawk_ooch_t* dat, hawk_oow_t len)
 | 
						|
{
 | 
						|
	sed_xtn_t* xtn = GET_XTN(s);
 | 
						|
 | 
						|
	Stream::Data iodata (xtn->sed, Stream::WRITE, arg);
 | 
						|
 | 
						|
	try
 | 
						|
	{
 | 
						|
		switch (cmd)
 | 
						|
		{
 | 
						|
			case HAWK_SED_IO_OPEN:
 | 
						|
				return xtn->sed->ostream->open(iodata);
 | 
						|
			case HAWK_SED_IO_CLOSE:
 | 
						|
				return xtn->sed->ostream->close(iodata);
 | 
						|
			case HAWK_SED_IO_WRITE:
 | 
						|
				return xtn->sed->ostream->write(iodata, dat, len);
 | 
						|
			default:
 | 
						|
				return -1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	catch (...)
 | 
						|
	{
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
const hawk_ooch_t* Sed::getErrorString (hawk_errnum_t num) const
 | 
						|
{
 | 
						|
	HAWK_ASSERT(this->dflerrstr != HAWK_NULL);
 | 
						|
	return this->dflerrstr(num);
 | 
						|
}
 | 
						|
 | 
						|
const hawk_ooch_t* Sed::xerrstr (hawk_sed_t* s, hawk_errnum_t num)
 | 
						|
{
 | 
						|
	sed_xtn_t* xtn = GET_XTN(s);
 | 
						|
	return xtn->sed->getErrorString(num);
 | 
						|
}
 | 
						|
 | 
						|
/////////////////////////////////
 | 
						|
HAWK_END_NAMESPACE(HAWK)
 | 
						|
/////////////////////////////////
 |