/******************************************************************************
  This file contains routines that can be bound to a Postgres backend and
  called by the backend in the process of processing queries.  The calling
  format for these routines is dictated by Postgres architecture.
******************************************************************************/

#include <stdio.h>
/* do not include libpq-fe.h for backend-loaded functions*/
/* #include "libpq-fe.h"  */
#include "postgres.h"
#include "utils/elog.h"
#include "utils/palloc.h"
#include "utils/mcxt.h"

typedef struct Complex
{
	double		x;
	double		y;
}			Complex;

/* These prototypes declare the requirements that Postgres places on these
   user written functions.
*/
Complex    *complex_in(char *str);
char	   *complex_out(Complex * complex);
Complex    *complex_add(Complex * a, Complex * b);
bool		complex_abs_lt(Complex * a, Complex * b);
bool		complex_abs_le(Complex * a, Complex * b);
bool		complex_abs_eq(Complex * a, Complex * b);
bool		complex_abs_ge(Complex * a, Complex * b);
bool		complex_abs_gt(Complex * a, Complex * b);
int4		complex_abs_cmp(Complex * a, Complex * b);


/*****************************************************************************
 * Input/Output functions
 *****************************************************************************/

Complex    *
complex_in(char *str)
{
	double		x,
				y;
	Complex    *result;

	if (sscanf(str, " ( %lf , %lf )", &x, &y) != 2)
	{
		elog(WARN, "complex_in: error in parsing \"%s\"", str);
		return NULL;
	}
	result = (Complex *) palloc(sizeof(Complex));
	result->x = x;
	result->y = y;
	return (result);
}

/*
 * You might have noticed a slight inconsistency between the following
 * declaration and the SQL definition:
 *	   CREATE FUNCTION complex_out(opaque) RETURNS opaque ...
 * The reason is that the argument pass into complex_out is really just a
 * pointer. POSTGRES thinks all output functions are:
 *	   char *out_func(char *);
 */
char	   *
complex_out(Complex * complex)
{
	char	   *result;

	if (complex == NULL)
		return (NULL);

	result = (char *) palloc(60);
	sprintf(result, "(%g,%g)", complex->x, complex->y);
	return (result);
}

/*****************************************************************************
 * New Operators
 *****************************************************************************/

Complex    *
complex_add(Complex * a, Complex * b)
{
	Complex    *result;

	result = (Complex *) palloc(sizeof(Complex));
	result->x = a->x + b->x;
	result->y = a->y + b->y;
	return (result);
}


/*****************************************************************************
 * Operator class for defining B-tree index
 *****************************************************************************/

#define Mag(c)	((c)->x*(c)->x + (c)->y*(c)->y)

bool
complex_abs_lt(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	return (amag < bmag);
}

bool
complex_abs_le(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	return (amag <= bmag);
}

bool
complex_abs_eq(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	return (amag == bmag);
}

bool
complex_abs_ge(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	return (amag >= bmag);
}

bool
complex_abs_gt(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	return (amag > bmag);
}

int4
complex_abs_cmp(Complex * a, Complex * b)
{
	double		amag = Mag(a),
				bmag = Mag(b);

	if (amag < bmag)
		return -1;
	else if (amag > bmag)
		return 1;
	else
		return 0;
}

/*****************************************************************************
 * test code
 *****************************************************************************/

/*
 * You should always test your code separately. Trust me, using POSTGRES to
 * debug your C function will be very painful and unproductive. In case of
 * POSTGRES crashing, it is impossible to tell whether the bug is in your
 * code or POSTGRES's.
 */
void		test_main(void);
void
test_main()
{
	Complex    *a;
	Complex    *b;

	a = complex_in("(4.01, 3.77 )");
	printf("a = %s\n", complex_out(a));
	b = complex_in("(1.0,2.0)");
	printf("b = %s\n", complex_out(b));
	printf("a +  b = %s\n", complex_out(complex_add(a, b)));
	printf("a <  b = %d\n", complex_abs_lt(a, b));
	printf("a <= b = %d\n", complex_abs_le(a, b));
	printf("a =  b = %d\n", complex_abs_eq(a, b));
	printf("a >= b = %d\n", complex_abs_ge(a, b));
	printf("a >  b = %d\n", complex_abs_gt(a, b));
}