File:  [parser3project] / parser3 / src / classes / math.C
Revision 1.2: download - view: text, annotated - select for diffs - revision graph
Fri Jul 6 11:13:35 2001 UTC (25 years ago) by parser
Branches: MAIN
CVS tags: HEAD
math 0

/** @file
	Parser: @b math parser class.

	Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com)

	Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf)
*/
static const char *RCSId="$Id: math.C,v 1.2 2001/07/06 11:13:35 parser Exp $"; 

#include "pa_config_includes.h"
#include "pa_common.h"
#include "pa_vint.h"
#include "pa_vmath.h"
#include "pa_request.h"

// defines

#define MATH_CLASS_NAME "math"

// class

class MMath : public Methoded {
public:
	MMath(Pool& pool);
public: // Methoded
	bool used_directly() { return true; }
};

// methods

static void _random(Request& r, const String& method_name, MethodParams *params) {
	Pool& pool=r.pool();

	Value& range=params->get_junction(0, "range must be expression");
    uint max=(uint)r.process(range).as_double();
    if(max<=1)
		PTHROW(0, 0,
			&method_name,
			"bad range [0..%u]", max);
	
	r.write_no_lang(*new(pool) VInt(pool, rand()%max));
}


typedef double (*math1_func_ptr)(double);
static double round(double param) { return floor(param+0.5); }
static double sign(double param) { return param > 0 ? 1 : ( param < 0 ? -1 : 0 ); }


static void math1(Request& r, 
				  const String& method_name, MethodParams *params,
				  math1_func_ptr func) {
	Pool& pool=r.pool();
	Value& param=params->get_junction(0, "parameter must be expression");

	Value& result=*new(pool) VDouble(pool, (*func)(r.process(param).as_double()));
	result.set_name(method_name);
	r.write_no_lang(result);
}

#define MATH1(name) \
	static void _##name(Request& r, const String& method_name, MethodParams *params) {\
		math1(r, method_name, params, &name);\
	}
#define MATH1P(name_parser, name_c) \
	static void _##name_parser(Request& r, const String& method_name, MethodParams *params) {\
		math1(r, method_name, params, &name_c);\
	}
MATH1(round);	MATH1(floor);	MATH1P(ceiling, ceil);
MATH1P(abs, fabs);	MATH1(sign);
MATH1(exp);	MATH1(log);	
MATH1(sin);	MATH1(asin);	
MATH1(cos);	MATH1(acos);	
MATH1(tan);	MATH1(atan);
MATH1(sqrt);


typedef double (*math2_func_ptr)(double, double);
static void math2(Request& r, 
				  const String& method_name, MethodParams *params,
				  math2_func_ptr func) {
	Pool& pool=r.pool();
	Value& a=params->get_junction(0, "parameter must be expression");
	Value& b=params->get_junction(1, "parameter must be expression");

	Value& result=*new(pool) VDouble(pool, (*func)(
		r.process(a).as_double(),
		r.process(b).as_double()));
	result.set_name(method_name);
	r.write_no_lang(result);
}

#define MATH2(name) \
	static void _##name(Request& r, const String& method_name, MethodParams *params) {\
		math2(r, method_name, params, &name);\
	}
MATH2(pow);

// constructor

MMath::MMath(Pool& apool) : Methoded(apool) {
	set_name(*NEW String(pool(), MATH_CLASS_NAME));


	// setting seed
	srand(getpid()+time(NULL));  rand();
	
	// ^FUNC(expr)	
#define ADD1(name) \
	add_native_method(#name, Method::CT_STATIC, _##name, 1, 1)

	ADD1(random);
	ADD1(round);	ADD1(floor);	ADD1(ceiling);
	ADD1(abs);	ADD1(sign);
	ADD1(exp);	ADD1(log);	
	ADD1(sin);	ADD1(asin);	
	ADD1(cos);	ADD1(acos);	
	ADD1(tan);	ADD1(atan);
	ADD1(sqrt);

#define ADD2(name) \
	add_native_method(#name, Method::CT_STATIC, _##name, 2, 2)

	// ^pow(x;y)
	ADD2(pow);

}

// global variables

Methoded *math_base_class;
Hash *math_consts;

// creator

Methoded *MMath_create(Pool& pool) {
	math_consts=new(pool) Hash(pool);
	math_consts->put(
		*new(pool) String(pool, "PI"), 
		new(pool) VDouble(pool, 3.1415926535));

	return math_base_class=new(pool) MMath(pool);
}

E-mail: