gambas-source-code/main/gbx/gbx_math.c
Benoît Minisini 4c02c6d338 ******** Merged /branches/64bits r918:1003 into /trunk
[CONFIGURATION]
* NEW: 64 bits port.

[EXAMPLES]
* BUG: Fixed the AnalogWatch example.

[WIKI CGI SCRIPT]
* NEW: Some little cosmetic changes.

[INTERPRETER]
* NEW: The extern function implementation has been redesigned and is now
  based on libffi, so that it works on 64 bits system. Because of a flaw in
  the compiler design, projects that use the Pointer datatype must be
  recompiled to be used on a 64 bits system. This flaw will be fixed in
  Gambas 3.
* OPT: Put some tables into read-only memory. About 1000 bytes are saved
  for each running interpreter, except the first one.
* BUG: Does not crash anymore if a component cannot be loaded.
* NEW: Spanish translation updated.
* NEW: A new interpreter API for returning a pointer.

[COMPILER]
* BUG: Correctly compiles LONG constants inside code.

[GB.DEBUG]
* BUG: Compiles and links the gb.debug components with the thread
  libraries.

[GB.DB.SQLITE3]
* BUG: Getting the primary index of a table without primary index is safe
  now.

[GB.GTK]
* BUG: Modified the GLib priority of watched descriptors, as the main loop 
  could enter in a loop in which user interface events were not managed.
* BUG: Message boxes use application title without crashing now.

[GB.OPENGL]
* BUG: Disable dead code.

[GB.QT.EXT]
* BUG: TextEdit.TextWidth and TextEdit.TextHeight were not declared as
  read-only properties.

[GB.XML.XSLT]
* BUG: XSLT class is now declared as being not creatable.


git-svn-id: svn://localhost/gambas/trunk@1006 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-01-17 21:39:26 +00:00

239 lines
3.8 KiB
C

/***************************************************************************
mathext.c
Mathematical extension routines
(c) 2000-2007 Benoit Minisini <gambas@users.sourceforge.net>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 1, or (at your option)
any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
***************************************************************************/
#define __GBX_MATH_C
#include <math.h>
#include <time.h>
#include <sys/time.h>
#include "gb_common.h"
#include "gbx_math.h"
/* This is a twisted generalized feedback shift register
that generates pseudo-random numbers.
Source code from the paper of Yann Guidon
in March edition of GNU/Linux Magazine France
*/
#define CRC32_STD 0x04C11DB7
#define GFSR_SIZE 15
static uint GFSR_table[GFSR_SIZE];
static uint GFSR_temp;
static uint GFSR_index;
static void GFSR_init(uint seed)
{
int i = 0, j;
uint t = seed;
do
{
t ^= (t >> 5) ^ (t << 1);
j = t >> 31;
t <<= 1;
if (j)
t ^= CRC32_STD;
GFSR_table[i++] = t;
}
while (i < GFSR_SIZE);
GFSR_temp = GFSR_table[GFSR_SIZE - 1];
GFSR_index = 0;
}
static uint GFSR_random(void)
{
int t;
GFSR_temp ^= GFSR_table[GFSR_index];
t = GFSR_temp;
GFSR_temp <<= 1;
if (t < 0)
GFSR_temp ^= CRC32_STD;
GFSR_table[GFSR_index] = GFSR_temp;
if (++GFSR_index >= GFSR_SIZE)
GFSR_index = 0;
return GFSR_temp;
}
//static long seed = 1;
double frac(double x)
{
x = fabs(x);
return x - floor(x);
}
int lsgn(int x)
{
return ((x > 0) ? 1 : ((x < 0) ? (-1) : 0));
}
int llsgn(int64_t x)
{
return ((x > 0) ? 1 : ((x < 0) ? (-1) : 0));
}
/*int64_t llabs(int64_t x)
{
return ((x < 0) ? (-x) : x);
}*/
int fsgn(double x)
{
return ((x > 0) ? 1 : ((x < 0) ? (-1) : 0));
}
double deg(double x)
{
return x * 180 / M_PI;
}
double rad(double x)
{
return x * M_PI / 180;
}
double fix(double x)
{
if (x >= 0)
return floor(x);
else
return -floor(fabs(x));
}
double frexp10(double x, int *exp)
{
long double l, f;
if (x == 0.0)
{
*exp = 0;
return x;
}
l = modfl(log10l(fabsl(x)), &f);
if (f == 0.0)
l = x;
else
l = powl(10, l);
if (x < 0.0) l = -l;
while (l >= 1.0)
{
l /= 10.0;
f++;
}
*exp = (int)f;
return l;
}
void randomize(bool set, uint seed)
{
struct timeval tv;
if (!set && gettimeofday(&tv, NULL) == 0)
seed = tv.tv_sec * 1000 + tv.tv_usec / 1000;
//GFSR_init(0x1A2B3C4D);
GFSR_init(seed);
}
double rnd(void)
{
/*seed = 16807L * (seed % 127773L) - 2836L * (seed / 127773L);
if (seed <= 0) seed += 2147483647;
return (double)seed / 2147483648.0;*/
uint64_t val;
val = GFSR_random();
val <<= 32;
val |= GFSR_random();
return (double)val / 0xFFFFFFFFFFFFFFFFULL;
}
double ang(double x, double y)
{
return atan2(y, x);
}
#if defined(OS_FREEBSD) || defined(OS_OPENBSD)
double exp10(double x)
{
return pow(10, x);
}
double log2(double x)
{
return log(x) / M_LN2;
}
#endif
#ifdef OS_OPENBSD
double exp2(double x)
{
return pow(2, x);
}
long double log10l(long double x)
{
return log10((double) x);
}
long double fabsl(long double x)
{
return fabs((double) x);
}
long double powl(long double x, long double y)
{
return pow((double) x, (double) y);
}
#endif
void MATH_init(void)
{
randomize(FALSE, 0);
}