181 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * $Id: awk04.c 523 2011-07-25 15:42:35Z hyunghwan.chung $
 | |
|  *
 | |
|     Copyright 2006-2011 Chung, Hyung-Hwan.
 | |
|     This file is part of QSE.
 | |
| 
 | |
|     QSE is free software: you can redistribute it and/or modify
 | |
|     it under the terms of the GNU Lesser General Public License as 
 | |
|     published by the Free Software Foundation, either version 3 of 
 | |
|     the License, or (at your option) any later version.
 | |
| 
 | |
|     QSE 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 Lesser General Public License for more details.
 | |
| 
 | |
|     You should have received a copy of the GNU Lesser General Public 
 | |
|     License along with QSE. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include <qse/awk/awk.h>
 | |
| #include <qse/awk/std.h>
 | |
| #include <qse/cmn/stdio.h>
 | |
| 
 | |
| static const qse_char_t* src = QSE_T(
 | |
| 	"function pow(x,y) { return x ** y; }"
 | |
| );
 | |
| 
 | |
| int main ()
 | |
| {
 | |
| 	qse_awk_t* awk = QSE_NULL;
 | |
| 	qse_awk_rtx_t* rtx = QSE_NULL;
 | |
| 	qse_awk_parsestd_in_t psin;
 | |
| 	qse_char_t* str;
 | |
| 	qse_size_t len;
 | |
| 	qse_awk_val_t* rtv = QSE_NULL;
 | |
| 	qse_awk_val_t* arg[2] = { QSE_NULL, QSE_NULL };
 | |
| 	qse_awk_fun_t* fun;
 | |
| 	int ret, i, opt;
 | |
| 
 | |
| 	qse_awk_rtx_valtostr_out_t out;
 | |
| 	qse_char_t numbuf[128];
 | |
| 
 | |
| 	/* create a main processor */
 | |
| 	awk = qse_awk_openstd (0);
 | |
| 	if (awk == QSE_NULL)  
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: cannot open awk\n"));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	opt = qse_awk_getoption(awk);
 | |
| 
 | |
| 	/* don't allow BEGIN, END, pattern-action blocks */
 | |
| 	opt &= ~QSE_AWK_PABLOCK;
 | |
| 	/* enable ** */
 | |
| 	opt |= QSE_AWK_EXTRAOPS;
 | |
| 
 | |
| 	qse_awk_setoption (awk, opt);
 | |
| 
 | |
| 	psin.type = QSE_AWK_PARSESTD_CP;
 | |
| 	psin.u.cp = src;
 | |
| 
 | |
| 	ret = qse_awk_parsestd (awk, &psin, QSE_NULL);
 | |
| 	if (ret == -1)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_geterrmsg(awk));
 | |
| 		goto oops;
 | |
| 	}
 | |
| 
 | |
| 	/* create a runtime context */
 | |
| 	rtx = qse_awk_rtx_openstd (
 | |
| 		awk, 
 | |
| 		0,
 | |
| 		QSE_T("awk04"),
 | |
| 		QSE_NULL, /* stdin */
 | |
| 		QSE_NULL  /* stdout */
 | |
| 	);
 | |
| 	if (rtx == QSE_NULL) 
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_geterrmsg(awk));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 	
 | |
| 	/* invoke the pow function */
 | |
| 	arg[0] = qse_awk_rtx_makeintval (rtx, 50);
 | |
| 	if (arg[0] == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 	qse_awk_rtx_refupval (rtx, arg[0]);
 | |
| 
 | |
| 	arg[1] = qse_awk_rtx_makeintval (rtx, 3);
 | |
| 	if (arg[1] == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 	qse_awk_rtx_refupval (rtx, arg[1]);
 | |
| 
 | |
| 	rtv = qse_awk_rtx_call (rtx, QSE_T("pow"), arg, 2);
 | |
| 	if (rtv == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	str = qse_awk_rtx_valtocpldup (rtx, rtv, &len);
 | |
| 	if (str == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	qse_printf (QSE_T("[%.*s]\n"), (int)len, str);
 | |
| 	qse_awk_rtx_freemem (rtx, str);
 | |
| 
 | |
| 	if (rtv) 
 | |
| 	{
 | |
| 		qse_awk_rtx_refdownval (rtx, rtv);
 | |
| 		rtv = QSE_NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* call the function again using different API functions */
 | |
| 	fun = qse_awk_rtx_findfun (rtx, QSE_T("pow"));
 | |
| 	if (fun == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	rtv = qse_awk_rtx_callfun (rtx, fun, arg, 2);
 | |
| 	if (rtv == QSE_NULL)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	/* Convert a value to a string in a different way */
 | |
| 	out.type = QSE_AWK_RTX_VALTOSTR_CPL;
 | |
| 	out.u.cpl.ptr = numbuf; /* used if the value is not a string */
 | |
| 	out.u.cpl.len = QSE_COUNTOF(numbuf);
 | |
| 	if (qse_awk_rtx_valtostr (rtx, rtv, &out) <= -1)
 | |
| 	{
 | |
| 		qse_fprintf (QSE_STDERR, QSE_T("error: %s\n"), 
 | |
| 			qse_awk_rtx_geterrmsg(rtx));
 | |
| 		ret = -1; goto oops;
 | |
| 	}
 | |
| 
 | |
| 	qse_printf (QSE_T("[%.*s]\n"), (int)out.u.cpl.len, out.u.cpl.ptr);
 | |
| 
 | |
| oops:
 | |
| 	/* clear the return value */
 | |
| 	if (rtv) 
 | |
| 	{
 | |
| 		qse_awk_rtx_refdownval (rtx, rtv);
 | |
| 		rtv = QSE_NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* dereference all arguments */
 | |
| 	for (i = 0; i < QSE_COUNTOF(arg); i++)
 | |
| 	{
 | |
| 		if (arg[i]) qse_awk_rtx_refdownval (rtx, arg[i]);
 | |
| 	}
 | |
| 
 | |
| 	/* destroy a runtime context */
 | |
| 	if (rtx) qse_awk_rtx_close (rtx);
 | |
| 	/* destroy the processor */
 | |
| 	if (awk) qse_awk_close (awk);
 | |
| 	return ret;
 | |
| }
 |