2007-06-18 23:26:00 +00:00
|
|
|
/*
|
2012-08-16 03:47:55 +00:00
|
|
|
* $Id$
|
2007-05-02 01:07:00 +00:00
|
|
|
*
|
2019-06-06 05:28:23 +00:00
|
|
|
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
2008-12-27 04:35:14 +00:00
|
|
|
|
2014-11-19 14:42:24 +00:00
|
|
|
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.
|
2008-12-27 04:35:14 +00:00
|
|
|
|
2014-11-19 14:42:24 +00:00
|
|
|
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.
|
2007-05-02 01:07:00 +00:00
|
|
|
*/
|
|
|
|
|
2016-04-29 03:55:42 +00:00
|
|
|
#include "awk-prv.h"
|
2019-04-18 08:42:54 +00:00
|
|
|
#include <qse/cmn/mbwc.h>
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2010-07-09 00:58:44 +00:00
|
|
|
static void free_fun (qse_htb_t* map, void* vptr, qse_size_t vlen)
|
2008-08-29 04:29:53 +00:00
|
|
|
{
|
2009-02-17 02:11:31 +00:00
|
|
|
qse_awk_t* awk = *(qse_awk_t**)QSE_XTN(map);
|
2009-02-01 03:59:46 +00:00
|
|
|
qse_awk_fun_t* f = (qse_awk_fun_t*)vptr;
|
2008-09-05 04:58:08 +00:00
|
|
|
|
|
|
|
/* f->name doesn't have to be freed */
|
2019-06-07 05:20:20 +00:00
|
|
|
/*qse_awk_freemem (awk, f->name);*/
|
2008-09-05 04:58:08 +00:00
|
|
|
|
2019-06-07 09:26:50 +00:00
|
|
|
if (f->argspec) qse_awk_freemem (awk, f->argspec);
|
2008-12-21 21:35:07 +00:00
|
|
|
qse_awk_clrpt (awk, f->body);
|
2019-06-07 05:20:20 +00:00
|
|
|
qse_awk_freemem (awk, f);
|
2008-08-29 04:29:53 +00:00
|
|
|
}
|
|
|
|
|
2010-07-09 00:58:44 +00:00
|
|
|
static void free_fnc (qse_htb_t* map, void* vptr, qse_size_t vlen)
|
2008-09-05 04:58:08 +00:00
|
|
|
{
|
2009-02-17 02:11:31 +00:00
|
|
|
qse_awk_t* awk = *(qse_awk_t**)QSE_XTN(map);
|
2009-02-01 03:59:46 +00:00
|
|
|
qse_awk_fnc_t* f = (qse_awk_fnc_t*)vptr;
|
2019-06-07 05:20:20 +00:00
|
|
|
qse_awk_freemem (awk, f);
|
2008-09-05 04:58:08 +00:00
|
|
|
}
|
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
static int init_token (qse_mmgr_t* mmgr, qse_awk_tok_t* tok)
|
2009-07-15 02:06:14 +00:00
|
|
|
{
|
2009-08-24 06:56:45 +00:00
|
|
|
tok->name = qse_str_open (mmgr, 0, 128);
|
|
|
|
if (tok->name == QSE_NULL) return -1;
|
2009-07-15 02:06:14 +00:00
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
tok->type = 0;
|
2010-08-18 07:15:14 +00:00
|
|
|
tok->loc.file = QSE_NULL;
|
|
|
|
tok->loc.line = 0;
|
|
|
|
tok->loc.colm = 0;
|
2009-07-15 02:06:14 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
static void fini_token (qse_awk_tok_t* tok)
|
2009-07-15 02:06:14 +00:00
|
|
|
{
|
2014-07-11 14:17:00 +00:00
|
|
|
if (tok->name)
|
2009-07-15 02:06:14 +00:00
|
|
|
{
|
2009-08-24 06:56:45 +00:00
|
|
|
qse_str_close (tok->name);
|
|
|
|
tok->name = QSE_NULL;
|
2009-07-15 02:06:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
static void clear_token (qse_awk_tok_t* tok)
|
2009-07-27 20:31:58 +00:00
|
|
|
{
|
2014-07-11 14:17:00 +00:00
|
|
|
if (tok->name) qse_str_clear (tok->name);
|
2009-08-24 06:56:45 +00:00
|
|
|
tok->type = 0;
|
2010-08-18 07:15:14 +00:00
|
|
|
tok->loc.file = QSE_NULL;
|
|
|
|
tok->loc.line = 0;
|
|
|
|
tok->loc.colm = 0;
|
2009-07-27 20:31:58 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
qse_awk_t* qse_awk_open (qse_mmgr_t* mmgr, qse_size_t xtnsize, const qse_awk_prm_t* prm, qse_awk_errnum_t* errnum)
|
2008-07-20 02:03:49 +00:00
|
|
|
{
|
2008-12-21 21:35:07 +00:00
|
|
|
qse_awk_t* awk;
|
2008-07-20 02:03:49 +00:00
|
|
|
|
2019-05-02 11:29:51 +00:00
|
|
|
awk = (qse_awk_t*)QSE_MMGR_ALLOC(mmgr, QSE_SIZEOF(qse_awk_t) + xtnsize);
|
2014-07-11 14:17:00 +00:00
|
|
|
if (awk)
|
|
|
|
{
|
|
|
|
int xret;
|
|
|
|
|
2019-05-02 11:29:51 +00:00
|
|
|
xret = qse_awk_init(awk, mmgr, prm);
|
2014-07-11 14:17:00 +00:00
|
|
|
if (xret <= -1)
|
|
|
|
{
|
|
|
|
if (errnum) *errnum = qse_awk_geterrnum(awk);
|
|
|
|
QSE_MMGR_FREE (mmgr, awk);
|
|
|
|
awk = QSE_NULL;
|
|
|
|
}
|
2019-06-24 14:24:14 +00:00
|
|
|
else QSE_MEMSET (QSE_XTN(awk), 0, xtnsize);
|
2014-07-11 14:17:00 +00:00
|
|
|
}
|
|
|
|
else if (errnum) *errnum = QSE_AWK_ENOMEM;
|
|
|
|
|
|
|
|
return awk;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qse_awk_close (qse_awk_t* awk)
|
|
|
|
{
|
|
|
|
qse_awk_fini (awk);
|
2019-06-24 08:53:49 +00:00
|
|
|
QSE_MMGR_FREE (awk->_mmgr, awk);
|
2014-07-11 14:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int qse_awk_init (qse_awk_t* awk, qse_mmgr_t* mmgr, const qse_awk_prm_t* prm)
|
|
|
|
{
|
2013-03-11 16:34:41 +00:00
|
|
|
static qse_htb_style_t treefuncbs =
|
2010-10-28 06:54:37 +00:00
|
|
|
{
|
|
|
|
{
|
|
|
|
QSE_HTB_COPIER_INLINE,
|
|
|
|
QSE_HTB_COPIER_DEFAULT
|
|
|
|
},
|
|
|
|
{
|
|
|
|
QSE_HTB_FREEER_DEFAULT,
|
|
|
|
free_fun
|
|
|
|
},
|
|
|
|
QSE_HTB_COMPER_DEFAULT,
|
|
|
|
QSE_HTB_KEEPER_DEFAULT,
|
2010-10-30 07:54:36 +00:00
|
|
|
QSE_HTB_SIZER_DEFAULT,
|
|
|
|
QSE_HTB_HASHER_DEFAULT
|
2010-10-28 06:54:37 +00:00
|
|
|
};
|
|
|
|
|
2013-03-11 16:34:41 +00:00
|
|
|
static qse_htb_style_t fncusercbs =
|
2010-10-28 06:54:37 +00:00
|
|
|
{
|
|
|
|
{
|
|
|
|
QSE_HTB_COPIER_INLINE,
|
|
|
|
QSE_HTB_COPIER_DEFAULT
|
|
|
|
},
|
|
|
|
{
|
|
|
|
QSE_HTB_FREEER_DEFAULT,
|
|
|
|
free_fnc
|
|
|
|
},
|
|
|
|
QSE_HTB_COMPER_DEFAULT,
|
|
|
|
QSE_HTB_KEEPER_DEFAULT,
|
2010-10-30 07:54:36 +00:00
|
|
|
QSE_HTB_SIZER_DEFAULT,
|
|
|
|
QSE_HTB_HASHER_DEFAULT
|
2010-10-28 06:54:37 +00:00
|
|
|
};
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
/* zero out the object */
|
2015-06-12 08:40:58 +00:00
|
|
|
QSE_MEMSET (awk, 0, QSE_SIZEOF(*awk));
|
2009-02-17 02:11:31 +00:00
|
|
|
|
|
|
|
/* remember the memory manager */
|
2019-06-24 08:53:49 +00:00
|
|
|
awk->_instsize = QSE_SIZEOF(*awk);
|
|
|
|
awk->_mmgr = mmgr;
|
|
|
|
awk->_cmgr = qse_getdflcmgr();
|
2009-02-17 02:11:31 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
/* initialize error handling fields */
|
|
|
|
awk->errinf.num = QSE_AWK_ENOERR;
|
|
|
|
awk->errinf.loc.line = 0;
|
|
|
|
awk->errinf.loc.colm = 0;
|
|
|
|
awk->errinf.loc.file = QSE_NULL;
|
|
|
|
awk->errstr = qse_awk_dflerrstr;
|
2019-11-12 07:55:32 +00:00
|
|
|
awk->haltall = 0;
|
2014-07-11 14:17:00 +00:00
|
|
|
|
2009-02-17 02:11:31 +00:00
|
|
|
/* progagate the primitive functions */
|
2011-05-18 08:37:51 +00:00
|
|
|
QSE_ASSERT (prm != QSE_NULL);
|
|
|
|
QSE_ASSERT (prm->math.pow != QSE_NULL);
|
2014-07-01 15:27:27 +00:00
|
|
|
QSE_ASSERT (prm->math.mod != QSE_NULL);
|
|
|
|
if (prm == QSE_NULL ||
|
|
|
|
prm->math.pow == QSE_NULL ||
|
|
|
|
prm->math.mod == QSE_NULL)
|
2009-02-27 04:56:12 +00:00
|
|
|
{
|
2014-07-11 14:17:00 +00:00
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_EINVAL, QSE_NULL);
|
|
|
|
goto oops;
|
2009-02-27 04:56:12 +00:00
|
|
|
}
|
2009-02-17 02:11:31 +00:00
|
|
|
awk->prm = *prm;
|
|
|
|
|
2019-04-18 08:42:54 +00:00
|
|
|
if (init_token(mmgr, &awk->ptok) <= -1 ||
|
|
|
|
init_token(mmgr, &awk->tok) <= -1 ||
|
|
|
|
init_token(mmgr, &awk->ntok) <= -1)
|
2014-07-11 14:17:00 +00:00
|
|
|
{
|
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_ENOMEM, QSE_NULL);
|
|
|
|
goto oops;
|
|
|
|
}
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
awk->opt.trait = QSE_AWK_MODERN;
|
|
|
|
#if defined(__OS2__) || defined(_WIN32) || defined(__DOS__)
|
|
|
|
awk->opt.trait |= QSE_AWK_CRLF;
|
|
|
|
#endif
|
2019-06-09 05:55:36 +00:00
|
|
|
awk->opt.rtx_stack_limit = QSE_AWK_DFL_RTX_STACK_LIMIT;
|
2009-02-01 03:59:46 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
awk->tree.ngbls = 0;
|
|
|
|
awk->tree.ngbls_base = 0;
|
|
|
|
awk->tree.begin = QSE_NULL;
|
|
|
|
awk->tree.begin_tail = QSE_NULL;
|
|
|
|
awk->tree.end = QSE_NULL;
|
|
|
|
awk->tree.end_tail = QSE_NULL;
|
|
|
|
awk->tree.chain = QSE_NULL;
|
|
|
|
awk->tree.chain_tail = QSE_NULL;
|
|
|
|
awk->tree.chain_size = 0;
|
|
|
|
|
|
|
|
/* TODO: initial map size?? */
|
|
|
|
awk->tree.funs = qse_htb_open (mmgr, QSE_SIZEOF(awk), 512, 70, QSE_SIZEOF(qse_char_t), 1);
|
|
|
|
awk->parse.funs = qse_htb_open (mmgr, QSE_SIZEOF(awk), 256, 70, QSE_SIZEOF(qse_char_t), 1);
|
|
|
|
awk->parse.named = qse_htb_open (mmgr, QSE_SIZEOF(awk), 256, 70, QSE_SIZEOF(qse_char_t), 1);
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2016-09-23 14:53:51 +00:00
|
|
|
awk->parse.gbls = qse_arr_open (mmgr, QSE_SIZEOF(awk), 128);
|
|
|
|
awk->parse.lcls = qse_arr_open (mmgr, QSE_SIZEOF(awk), 64);
|
|
|
|
awk->parse.params = qse_arr_open (mmgr, QSE_SIZEOF(awk), 32);
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
awk->fnc.sys = QSE_NULL;
|
|
|
|
awk->fnc.user = qse_htb_open (mmgr, QSE_SIZEOF(awk), 512, 70, QSE_SIZEOF(qse_char_t), 1);
|
|
|
|
awk->modtab = qse_rbt_open (mmgr, 0, QSE_SIZEOF(qse_char_t), 1);
|
|
|
|
|
|
|
|
if (awk->tree.funs == QSE_NULL ||
|
|
|
|
awk->parse.funs == QSE_NULL ||
|
|
|
|
awk->parse.named == QSE_NULL ||
|
|
|
|
awk->parse.gbls == QSE_NULL ||
|
2009-02-02 08:28:04 +00:00
|
|
|
awk->parse.lcls == QSE_NULL ||
|
2014-07-11 14:17:00 +00:00
|
|
|
awk->parse.params == QSE_NULL ||
|
|
|
|
awk->fnc.user == QSE_NULL ||
|
|
|
|
awk->modtab == QSE_NULL)
|
|
|
|
{
|
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_ENOMEM, QSE_NULL);
|
|
|
|
goto oops;
|
|
|
|
}
|
|
|
|
|
|
|
|
*(qse_awk_t**)QSE_XTN(awk->tree.funs) = awk;
|
|
|
|
qse_htb_setstyle (awk->tree.funs, &treefuncbs);
|
|
|
|
|
|
|
|
*(qse_awk_t**)QSE_XTN(awk->parse.funs) = awk;
|
|
|
|
qse_htb_setstyle (awk->parse.funs, qse_gethtbstyle(QSE_HTB_STYLE_INLINE_KEY_COPIER));
|
|
|
|
|
|
|
|
*(qse_awk_t**)QSE_XTN(awk->parse.named) = awk;
|
|
|
|
qse_htb_setstyle (awk->parse.named, qse_gethtbstyle(QSE_HTB_STYLE_INLINE_KEY_COPIER));
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2009-02-17 02:11:31 +00:00
|
|
|
*(qse_awk_t**)QSE_XTN(awk->parse.gbls) = awk;
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_setscale (awk->parse.gbls, QSE_SIZEOF(qse_char_t));
|
|
|
|
qse_arr_setcopier (awk->parse.gbls, QSE_ARR_COPIER_INLINE);
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2009-02-17 02:11:31 +00:00
|
|
|
*(qse_awk_t**)QSE_XTN(awk->parse.lcls) = awk;
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_setscale (awk->parse.lcls, QSE_SIZEOF(qse_char_t));
|
|
|
|
qse_arr_setcopier (awk->parse.lcls, QSE_ARR_COPIER_INLINE);
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2009-02-17 02:11:31 +00:00
|
|
|
*(qse_awk_t**)QSE_XTN(awk->parse.params) = awk;
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_setscale (awk->parse.params, QSE_SIZEOF(qse_char_t));
|
|
|
|
qse_arr_setcopier (awk->parse.params, QSE_ARR_COPIER_INLINE);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2009-02-17 02:11:31 +00:00
|
|
|
*(qse_awk_t**)QSE_XTN(awk->fnc.user) = awk;
|
2013-03-11 16:34:41 +00:00
|
|
|
qse_htb_setstyle (awk->fnc.user, &fncusercbs);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
qse_rbt_setstyle (awk->modtab, qse_getrbtstyle(QSE_RBT_STYLE_INLINE_COPIERS));
|
2012-10-22 09:36:15 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
if (qse_awk_initgbls (awk) <= -1)
|
|
|
|
{
|
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_ENOMEM, QSE_NULL);
|
|
|
|
goto oops;
|
|
|
|
}
|
2007-09-25 00:12:00 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
return 0;
|
2008-10-11 05:16:31 +00:00
|
|
|
|
|
|
|
oops:
|
2012-10-22 09:36:15 +00:00
|
|
|
if (awk->modtab) qse_rbt_close (awk->modtab);
|
2010-07-09 00:58:44 +00:00
|
|
|
if (awk->fnc.user) qse_htb_close (awk->fnc.user);
|
2016-09-23 14:53:51 +00:00
|
|
|
if (awk->parse.params) qse_arr_close (awk->parse.params);
|
|
|
|
if (awk->parse.lcls) qse_arr_close (awk->parse.lcls);
|
|
|
|
if (awk->parse.gbls) qse_arr_close (awk->parse.gbls);
|
2010-07-09 00:58:44 +00:00
|
|
|
if (awk->parse.named) qse_htb_close (awk->parse.named);
|
|
|
|
if (awk->parse.funs) qse_htb_close (awk->parse.funs);
|
|
|
|
if (awk->tree.funs) qse_htb_close (awk->tree.funs);
|
2009-08-24 06:56:45 +00:00
|
|
|
fini_token (&awk->ntok);
|
|
|
|
fini_token (&awk->tok);
|
|
|
|
fini_token (&awk->ptok);
|
2008-12-21 21:35:07 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
return -1;
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
void qse_awk_fini (qse_awk_t* awk)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2012-09-07 15:13:55 +00:00
|
|
|
qse_awk_ecb_t* ecb;
|
2012-11-02 14:08:46 +00:00
|
|
|
int i;
|
2012-09-07 15:13:55 +00:00
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
qse_awk_clear (awk);
|
2009-02-01 03:59:46 +00:00
|
|
|
/*qse_awk_clrfnc (awk);*/
|
2012-09-07 15:13:55 +00:00
|
|
|
|
|
|
|
for (ecb = awk->ecb; ecb; ecb = ecb->next)
|
|
|
|
if (ecb->close) ecb->close (awk);
|
|
|
|
|
2012-10-22 09:36:15 +00:00
|
|
|
qse_rbt_close (awk->modtab);
|
2010-07-09 00:58:44 +00:00
|
|
|
qse_htb_close (awk->fnc.user);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_close (awk->parse.params);
|
|
|
|
qse_arr_close (awk->parse.lcls);
|
|
|
|
qse_arr_close (awk->parse.gbls);
|
2010-07-09 00:58:44 +00:00
|
|
|
qse_htb_close (awk->parse.named);
|
|
|
|
qse_htb_close (awk->parse.funs);
|
2007-12-12 08:05:58 +00:00
|
|
|
|
2010-07-09 00:58:44 +00:00
|
|
|
qse_htb_close (awk->tree.funs);
|
2009-08-07 02:27:14 +00:00
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
fini_token (&awk->ntok);
|
|
|
|
fini_token (&awk->tok);
|
|
|
|
fini_token (&awk->ptok);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2019-05-12 17:03:30 +00:00
|
|
|
if (awk->parse.incl_hist.ptr) qse_awk_freemem (awk, awk->parse.incl_hist.ptr);
|
2013-05-15 12:05:24 +00:00
|
|
|
qse_awk_clearsionames (awk);
|
|
|
|
|
2012-10-22 09:36:15 +00:00
|
|
|
/* destroy dynamically allocated options */
|
2012-11-02 14:08:46 +00:00
|
|
|
for (i = 0; i < QSE_COUNTOF(awk->opt.mod); i++)
|
|
|
|
{
|
2019-06-07 05:20:20 +00:00
|
|
|
if (awk->opt.mod[i].ptr) qse_awk_freemem (awk, awk->opt.mod[i].ptr);
|
2012-11-02 14:08:46 +00:00
|
|
|
}
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2012-10-22 09:36:15 +00:00
|
|
|
static qse_rbt_walk_t unload_module (qse_rbt_t* rbt, qse_rbt_pair_t* pair, void* ctx)
|
|
|
|
{
|
|
|
|
qse_awk_t* awk = (qse_awk_t*)ctx;
|
|
|
|
qse_awk_mod_data_t* md;
|
|
|
|
|
|
|
|
md = QSE_RBT_VPTR(pair);
|
2012-10-23 17:08:56 +00:00
|
|
|
if (md->mod.unload) md->mod.unload (&md->mod, awk);
|
2013-02-01 14:30:25 +00:00
|
|
|
if (md->handle) awk->prm.modclose (awk, md->handle);
|
2012-10-22 09:36:15 +00:00
|
|
|
|
|
|
|
return QSE_RBT_WALK_FORWARD;
|
|
|
|
}
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
void qse_awk_clear (qse_awk_t* awk)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2012-09-07 15:13:55 +00:00
|
|
|
qse_awk_ecb_t* ecb;
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
for (ecb = awk->ecb; ecb; ecb = ecb->next)
|
2019-06-09 05:55:36 +00:00
|
|
|
{
|
2014-07-11 14:17:00 +00:00
|
|
|
if (ecb->clear) ecb->clear (awk);
|
2019-06-09 05:55:36 +00:00
|
|
|
}
|
2012-09-07 15:13:55 +00:00
|
|
|
|
2019-11-12 07:55:32 +00:00
|
|
|
awk->haltall = 0;
|
2007-10-12 00:13:00 +00:00
|
|
|
|
2009-08-24 06:56:45 +00:00
|
|
|
clear_token (&awk->tok);
|
|
|
|
clear_token (&awk->ntok);
|
|
|
|
clear_token (&awk->ptok);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2015-03-12 06:39:16 +00:00
|
|
|
/* clear all loaded modules */
|
2012-10-22 09:36:15 +00:00
|
|
|
qse_rbt_walk (awk->modtab, unload_module, awk);
|
|
|
|
qse_rbt_clear (awk->modtab);
|
|
|
|
|
2016-09-23 14:53:51 +00:00
|
|
|
QSE_ASSERT (QSE_ARR_SIZE(awk->parse.gbls) == awk->tree.ngbls);
|
2008-10-11 05:16:31 +00:00
|
|
|
/* delete all non-builtin global variables */
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_delete (
|
2009-02-03 03:54:32 +00:00
|
|
|
awk->parse.gbls, awk->tree.ngbls_base,
|
2016-09-23 14:53:51 +00:00
|
|
|
QSE_ARR_SIZE(awk->parse.gbls) - awk->tree.ngbls_base);
|
2007-09-25 20:25:00 +00:00
|
|
|
|
2016-09-23 14:53:51 +00:00
|
|
|
qse_arr_clear (awk->parse.lcls);
|
|
|
|
qse_arr_clear (awk->parse.params);
|
2010-07-09 00:58:44 +00:00
|
|
|
qse_htb_clear (awk->parse.named);
|
|
|
|
qse_htb_clear (awk->parse.funs);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2009-02-02 08:28:04 +00:00
|
|
|
awk->parse.nlcls_max = 0;
|
2012-10-21 16:19:03 +00:00
|
|
|
awk->parse.depth.block = 0;
|
|
|
|
awk->parse.depth.loop = 0;
|
|
|
|
awk->parse.depth.expr = 0;
|
|
|
|
awk->parse.depth.incl = 0;
|
2019-06-09 05:55:36 +00:00
|
|
|
awk->parse.pragma.trait = (awk->opt.trait & QSE_AWK_IMPLICIT);
|
|
|
|
awk->parse.pragma.rtx_stack_limit = 0;
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2019-05-12 17:03:30 +00:00
|
|
|
awk->parse.incl_hist.count =0;
|
|
|
|
|
2019-04-30 07:23:34 +00:00
|
|
|
/* clear parse trees */
|
2009-02-03 03:54:32 +00:00
|
|
|
/*awk->tree.ngbls_base = 0;
|
2019-04-30 07:23:34 +00:00
|
|
|
awk->tree.ngbls = 0; */
|
2009-02-03 03:54:32 +00:00
|
|
|
awk->tree.ngbls = awk->tree.ngbls_base;
|
2007-09-25 20:25:00 +00:00
|
|
|
|
2009-02-01 03:59:46 +00:00
|
|
|
awk->tree.cur_fun.ptr = QSE_NULL;
|
|
|
|
awk->tree.cur_fun.len = 0;
|
2010-07-09 00:58:44 +00:00
|
|
|
qse_htb_clear (awk->tree.funs);
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2019-05-12 17:03:30 +00:00
|
|
|
if (awk->tree.begin)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2008-12-21 21:35:07 +00:00
|
|
|
/*QSE_ASSERT (awk->tree.begin->next == QSE_NULL);*/
|
|
|
|
qse_awk_clrpt (awk, awk->tree.begin);
|
|
|
|
awk->tree.begin = QSE_NULL;
|
2019-04-30 07:23:34 +00:00
|
|
|
awk->tree.begin_tail = QSE_NULL;
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2019-05-12 17:03:30 +00:00
|
|
|
if (awk->tree.end)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2008-12-21 21:35:07 +00:00
|
|
|
/*QSE_ASSERT (awk->tree.end->next == QSE_NULL);*/
|
|
|
|
qse_awk_clrpt (awk, awk->tree.end);
|
|
|
|
awk->tree.end = QSE_NULL;
|
2019-04-30 07:23:34 +00:00
|
|
|
awk->tree.end_tail = QSE_NULL;
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2019-05-12 17:03:30 +00:00
|
|
|
while (awk->tree.chain)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2008-12-21 21:35:07 +00:00
|
|
|
qse_awk_chain_t* next = awk->tree.chain->next;
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2008-12-21 21:35:07 +00:00
|
|
|
if (awk->tree.chain->pattern != QSE_NULL)
|
|
|
|
qse_awk_clrpt (awk, awk->tree.chain->pattern);
|
|
|
|
if (awk->tree.chain->action != QSE_NULL)
|
|
|
|
qse_awk_clrpt (awk, awk->tree.chain->action);
|
2019-06-07 05:20:20 +00:00
|
|
|
qse_awk_freemem (awk, awk->tree.chain);
|
2007-05-02 01:07:00 +00:00
|
|
|
awk->tree.chain = next;
|
|
|
|
}
|
|
|
|
|
2014-07-11 14:17:00 +00:00
|
|
|
awk->tree.chain_tail = QSE_NULL;
|
2007-05-02 01:07:00 +00:00
|
|
|
awk->tree.chain_size = 0;
|
|
|
|
|
2009-08-26 03:50:07 +00:00
|
|
|
/* this table must not be cleared here as there can be a reference
|
2013-05-15 12:05:24 +00:00
|
|
|
* to an entry of this table from errinf.loc.file when qse_awk_parse()
|
2009-08-26 03:50:07 +00:00
|
|
|
* failed. this table is cleared in qse_awk_parse().
|
2013-05-15 12:05:24 +00:00
|
|
|
* qse_awk_claersionames (awk);
|
2009-08-26 03:50:07 +00:00
|
|
|
*/
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2013-01-05 14:38:30 +00:00
|
|
|
void qse_awk_getprm (qse_awk_t* awk, qse_awk_prm_t* prm)
|
2008-11-27 03:05:00 +00:00
|
|
|
{
|
2013-01-05 14:38:30 +00:00
|
|
|
*prm = awk->prm;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qse_awk_setprm (qse_awk_t* awk, const qse_awk_prm_t* prm)
|
|
|
|
{
|
|
|
|
awk->prm = *prm;
|
2008-11-27 03:05:00 +00:00
|
|
|
}
|
|
|
|
|
2014-07-08 14:30:42 +00:00
|
|
|
static int dup_str_opt (qse_awk_t* awk, const void* value, qse_cstr_t* tmp)
|
2013-01-22 14:20:10 +00:00
|
|
|
{
|
|
|
|
if (value)
|
|
|
|
{
|
2019-06-24 08:53:49 +00:00
|
|
|
tmp->ptr = qse_strdup(value, qse_awk_getmmgr(awk));
|
2019-04-30 07:23:34 +00:00
|
|
|
if (!tmp->ptr)
|
2013-01-22 14:20:10 +00:00
|
|
|
{
|
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_ENOMEM, QSE_NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
2019-04-30 07:23:34 +00:00
|
|
|
tmp->len = qse_strlen(tmp->ptr);
|
2013-01-22 14:20:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tmp->ptr = QSE_NULL;
|
|
|
|
tmp->len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-10-21 16:19:03 +00:00
|
|
|
int qse_awk_setopt (qse_awk_t* awk, qse_awk_opt_t id, const void* value)
|
2007-05-02 01:07:00 +00:00
|
|
|
{
|
2012-10-21 16:19:03 +00:00
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case QSE_AWK_TRAIT:
|
|
|
|
awk->opt.trait = *(const int*)value;
|
|
|
|
return 0;
|
|
|
|
|
2012-11-02 14:08:46 +00:00
|
|
|
case QSE_AWK_MODPREFIX:
|
|
|
|
case QSE_AWK_MODPOSTFIX:
|
2012-10-22 09:36:15 +00:00
|
|
|
{
|
2014-07-08 14:30:42 +00:00
|
|
|
qse_cstr_t tmp;
|
2012-11-02 14:08:46 +00:00
|
|
|
int idx;
|
2012-10-22 09:36:15 +00:00
|
|
|
|
2013-01-22 14:20:10 +00:00
|
|
|
if (dup_str_opt (awk, value, &tmp) <= -1) return -1;
|
|
|
|
|
2012-11-02 14:08:46 +00:00
|
|
|
idx = id - QSE_AWK_MODPREFIX;
|
2019-06-07 05:20:20 +00:00
|
|
|
if (awk->opt.mod[idx].ptr) qse_awk_freemem (awk, awk->opt.mod[idx].ptr);
|
2012-10-22 09:36:15 +00:00
|
|
|
|
2012-11-02 14:08:46 +00:00
|
|
|
awk->opt.mod[idx] = tmp;
|
2012-10-22 09:36:15 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-01-22 14:20:10 +00:00
|
|
|
case QSE_AWK_INCLUDEDIRS:
|
|
|
|
{
|
2014-07-08 14:30:42 +00:00
|
|
|
qse_cstr_t tmp;
|
2013-01-22 14:20:10 +00:00
|
|
|
if (dup_str_opt (awk, value, &tmp) <= -1) return -1;
|
2019-06-07 05:20:20 +00:00
|
|
|
if (awk->opt.incldirs.ptr) qse_awk_freemem (awk, awk->opt.incldirs.ptr);
|
2013-01-22 14:20:10 +00:00
|
|
|
awk->opt.incldirs = tmp;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-10-21 16:19:03 +00:00
|
|
|
case QSE_AWK_DEPTH_INCLUDE:
|
|
|
|
case QSE_AWK_DEPTH_BLOCK_PARSE:
|
|
|
|
case QSE_AWK_DEPTH_BLOCK_RUN:
|
|
|
|
case QSE_AWK_DEPTH_EXPR_PARSE:
|
|
|
|
case QSE_AWK_DEPTH_EXPR_RUN:
|
|
|
|
case QSE_AWK_DEPTH_REX_BUILD:
|
|
|
|
case QSE_AWK_DEPTH_REX_MATCH:
|
|
|
|
awk->opt.depth.a[id - QSE_AWK_DEPTH_INCLUDE] = *(const qse_size_t*)value;
|
|
|
|
return 0;
|
2019-06-09 05:55:36 +00:00
|
|
|
|
|
|
|
case QSE_AWK_RTX_STACK_LIMIT:
|
|
|
|
awk->opt.rtx_stack_limit = *(const qse_size_t*)value;
|
2019-06-10 04:57:51 +00:00
|
|
|
if (awk->opt.rtx_stack_limit < QSE_AWK_MIN_RTX_STACK_LIMIT) awk->opt.rtx_stack_limit = QSE_AWK_MIN_RTX_STACK_LIMIT;
|
|
|
|
else if (awk->opt.rtx_stack_limit > QSE_AWK_MAX_RTX_STACK_LIMIT) awk->opt.rtx_stack_limit = QSE_AWK_MAX_RTX_STACK_LIMIT;
|
2019-06-09 05:55:36 +00:00
|
|
|
return 0;
|
2012-10-21 16:19:03 +00:00
|
|
|
}
|
2007-05-02 01:07:00 +00:00
|
|
|
|
2012-10-21 16:19:03 +00:00
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_EINVAL, QSE_NULL);
|
|
|
|
return -1;
|
2007-05-02 01:07:00 +00:00
|
|
|
}
|
|
|
|
|
2013-01-30 14:08:38 +00:00
|
|
|
int qse_awk_getopt (qse_awk_t* awk, qse_awk_opt_t id, void* value)
|
2007-10-12 00:13:00 +00:00
|
|
|
{
|
2012-10-21 16:19:03 +00:00
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case QSE_AWK_TRAIT:
|
|
|
|
*(int*)value = awk->opt.trait;
|
|
|
|
return 0;
|
|
|
|
|
2012-11-02 14:08:46 +00:00
|
|
|
case QSE_AWK_MODPREFIX:
|
|
|
|
case QSE_AWK_MODPOSTFIX:
|
|
|
|
*(const qse_char_t**)value = awk->opt.mod[id - QSE_AWK_MODPREFIX].ptr;
|
2012-10-22 09:36:15 +00:00
|
|
|
return 0;
|
|
|
|
|
2013-01-22 14:20:10 +00:00
|
|
|
case QSE_AWK_INCLUDEDIRS:
|
|
|
|
*(const qse_char_t**)value = awk->opt.incldirs.ptr;
|
|
|
|
return 0;
|
|
|
|
|
2012-10-21 16:19:03 +00:00
|
|
|
case QSE_AWK_DEPTH_INCLUDE:
|
|
|
|
case QSE_AWK_DEPTH_BLOCK_PARSE:
|
|
|
|
case QSE_AWK_DEPTH_BLOCK_RUN:
|
|
|
|
case QSE_AWK_DEPTH_EXPR_PARSE:
|
|
|
|
case QSE_AWK_DEPTH_EXPR_RUN:
|
|
|
|
case QSE_AWK_DEPTH_REX_BUILD:
|
|
|
|
case QSE_AWK_DEPTH_REX_MATCH:
|
|
|
|
*(qse_size_t*)value = awk->opt.depth.a[id - QSE_AWK_DEPTH_INCLUDE];
|
|
|
|
return 0;
|
2019-06-09 05:55:36 +00:00
|
|
|
|
|
|
|
case QSE_AWK_RTX_STACK_LIMIT:
|
|
|
|
*(qse_size_t*)value = awk->opt.rtx_stack_limit;
|
|
|
|
return 0;
|
2012-10-21 16:19:03 +00:00
|
|
|
};
|
2007-11-12 00:07:00 +00:00
|
|
|
|
2012-10-21 16:19:03 +00:00
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_EINVAL, QSE_NULL);
|
|
|
|
return -1;
|
2009-08-07 02:27:14 +00:00
|
|
|
}
|
|
|
|
|
2019-11-12 07:55:32 +00:00
|
|
|
void qse_awk_haltall (qse_awk_t* awk)
|
2009-08-07 02:27:14 +00:00
|
|
|
{
|
2019-11-12 07:55:32 +00:00
|
|
|
awk->haltall = 1;
|
2012-10-21 16:19:03 +00:00
|
|
|
qse_awk_seterrnum (awk, QSE_AWK_EINVAL, QSE_NULL);
|
2009-08-07 02:27:14 +00:00
|
|
|
}
|
2012-09-07 15:13:55 +00:00
|
|
|
|
|
|
|
qse_awk_ecb_t* qse_awk_popecb (qse_awk_t* awk)
|
|
|
|
{
|
|
|
|
qse_awk_ecb_t* top = awk->ecb;
|
|
|
|
if (top) awk->ecb = top->next;
|
|
|
|
return top;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qse_awk_pushecb (qse_awk_t* awk, qse_awk_ecb_t* ecb)
|
|
|
|
{
|
|
|
|
ecb->next = awk->ecb;
|
|
|
|
awk->ecb = ecb;
|
|
|
|
}
|
|
|
|
|