File:  [parser3project] / parser3 / src / classes / math.C
Revision 1.8: download - view: text, annotated - select for diffs - revision graph
Thu Nov 1 16:41:52 2001 UTC (24 years, 8 months ago) by paf
Branches: MAIN
CVS tags: HEAD
HAVE_TRUNC HAVE_ROUND checks [for hp, there are such in math.h]

/** @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)

	$Id: math.C,v 1.8 2001/11/01 16:41:52 paf Exp $
*/

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

#ifdef WIN32
#	include <windows.h>
#endif

// defines

#define PI 3.1415926535
#define MATH_CLASS_NAME "math"

// class

class MMath : public Methoded {
public:
	MMath(Pool& pool);
	void configure_admin(Request& r);

public: // Methoded
	bool used_directly() { return true; }
};

// methods
static unsigned int randomizer=0;
static void _random(Request& r, const String& method_name, MethodParams *params) {
	Pool& pool=r.pool();

	Value& range=params->as_junction(0, "range must be expression");
    uint max=(uint)r.process(range).as_double();
    if(max<=1)
		throw Exception(0, 0,
			&method_name,
			"bad range [0..%u]", max);
	
	Value& result=*new(pool) VInt(pool, (int)(
		((double)((randomizer=rand())% RAND_MAX)) / RAND_MAX * (max + 1)));
	result.set_name(method_name);
	r.write_no_lang(result);
}


typedef double (*math1_func_ptr)(double);
static double frac(double param) { return param-trunc(param); }
static double degrees(double param) { return param /PI *180; }
static double radians(double param) { return param /180 *PI; }

static void math1(Request& r, 
				  const String& method_name, MethodParams *params,
				  math1_func_ptr func) {
	Pool& pool=r.pool();
	Value& param=params->as_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);
MATH1(trunc);	MATH1(frac);
MATH1P(abs, fabs);	MATH1(sign);
MATH1(exp);	MATH1(log);	
MATH1(sin);	MATH1(asin);	
MATH1(cos);	MATH1(acos);	
MATH1(tan);	MATH1(atan);
MATH1(degrees);	MATH1(radians);
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->as_junction(0, "parameter must be expression");
	Value& b=params->as_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));
	
	// ^FUNC(expr)	
#define ADD1(name) \
	add_native_method(#name, Method::CT_STATIC, _##name, 1, 1)

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

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

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

}

// in MSVC each thread has it's own pseudo-random sequence
// in win32 apache each thread can handle multiple requests
// so to get proper randoms we remember random generated in one thread
void MMath::configure_admin(Request&) {
	// setting seed
	srand(
		randomizer ^
#ifdef WIN32
		GetCurrentThreadId() ^
#else
		getpid() ^
#endif
		(unsigned int)time(NULL)
	);
	if(!randomizer)
		randomizer=rand();
}

// 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, PI));

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

E-mail: