[GRASS-SVN] r62435 - in grass/trunk/lib: driver gpde ogsf

svn_grass at osgeo.org svn_grass at osgeo.org
Tue Oct 28 06:30:41 PDT 2014


Author: martinl
Date: 2014-10-28 06:30:41 -0700 (Tue, 28 Oct 2014)
New Revision: 62435

Added:
   grass/trunk/lib/driver/box.c
   grass/trunk/lib/driver/color.c
   grass/trunk/lib/driver/draw.c
   grass/trunk/lib/driver/erase.c
   grass/trunk/lib/driver/font.c
   grass/trunk/lib/driver/get_t_box.c
   grass/trunk/lib/driver/graph.c
   grass/trunk/lib/driver/line_width.c
   grass/trunk/lib/driver/move.c
   grass/trunk/lib/driver/raster.c
   grass/trunk/lib/driver/set_window.c
   grass/trunk/lib/driver/text.c
   grass/trunk/lib/driver/text_size.c
   grass/trunk/lib/gpde/n_arrays.c
   grass/trunk/lib/gpde/n_arrays_calc.c
   grass/trunk/lib/gpde/n_arrays_io.c
   grass/trunk/lib/gpde/n_geom.c
   grass/trunk/lib/gpde/n_gradient.c
   grass/trunk/lib/gpde/n_gradient_calc.c
   grass/trunk/lib/gpde/n_gwflow.c
   grass/trunk/lib/gpde/n_heatflow.c
   grass/trunk/lib/gpde/n_les.c
   grass/trunk/lib/gpde/n_les_assemble.c
   grass/trunk/lib/gpde/n_parse_options.c
   grass/trunk/lib/gpde/n_solute_transport.c
   grass/trunk/lib/gpde/n_tools.c
   grass/trunk/lib/gpde/n_upwind.c
   grass/trunk/lib/ogsf/gk2.c
   grass/trunk/lib/ogsf/gp2.c
   grass/trunk/lib/ogsf/gp3.c
   grass/trunk/lib/ogsf/gs2.c
   grass/trunk/lib/ogsf/gs3.c
   grass/trunk/lib/ogsf/gs_util.c
   grass/trunk/lib/ogsf/gsx.c
   grass/trunk/lib/ogsf/gv2.c
   grass/trunk/lib/ogsf/gv3.c
   grass/trunk/lib/ogsf/gvl2.c
   grass/trunk/lib/ogsf/gvl3.c
Removed:
   grass/trunk/lib/driver/Box.c
   grass/trunk/lib/driver/Color.c
   grass/trunk/lib/driver/Draw.c
   grass/trunk/lib/driver/Erase.c
   grass/trunk/lib/driver/Font.c
   grass/trunk/lib/driver/Get_t_box.c
   grass/trunk/lib/driver/Graph.c
   grass/trunk/lib/driver/Line_width.c
   grass/trunk/lib/driver/Move.c
   grass/trunk/lib/driver/Raster.c
   grass/trunk/lib/driver/Set_window.c
   grass/trunk/lib/driver/Text.c
   grass/trunk/lib/driver/Text_size.c
   grass/trunk/lib/gpde/N_arrays.c
   grass/trunk/lib/gpde/N_arrays_calc.c
   grass/trunk/lib/gpde/N_arrays_io.c
   grass/trunk/lib/gpde/N_geom.c
   grass/trunk/lib/gpde/N_gradient.c
   grass/trunk/lib/gpde/N_gradient_calc.c
   grass/trunk/lib/gpde/N_gwflow.c
   grass/trunk/lib/gpde/N_heatflow.c
   grass/trunk/lib/gpde/N_les.c
   grass/trunk/lib/gpde/N_les_assemble.c
   grass/trunk/lib/gpde/N_parse_options.c
   grass/trunk/lib/gpde/N_solute_transport.c
   grass/trunk/lib/gpde/N_tools.c
   grass/trunk/lib/gpde/N_upwind.c
   grass/trunk/lib/ogsf/GK2.c
   grass/trunk/lib/ogsf/GP2.c
   grass/trunk/lib/ogsf/GS2.c
   grass/trunk/lib/ogsf/GSX.c
   grass/trunk/lib/ogsf/GS_util.c
   grass/trunk/lib/ogsf/GV2.c
   grass/trunk/lib/ogsf/GVL2.c
   grass/trunk/lib/ogsf/Gp3.c
   grass/trunk/lib/ogsf/Gs3.c
   grass/trunk/lib/ogsf/Gv3.c
   grass/trunk/lib/ogsf/Gvl3.c
Log:
lib: rename files to lowercase


Deleted: grass/trunk/lib/driver/Box.c
===================================================================
--- grass/trunk/lib/driver/Box.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Box.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,9 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Box_abs(double x1, double y1, double x2, double y2)
-{
-    if (driver->Box)
-	(*driver->Box)(x1, y1, x2, y2);
-}
-

Deleted: grass/trunk/lib/driver/Color.c
===================================================================
--- grass/trunk/lib/driver/Color.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Color.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,20 +0,0 @@
-#include <grass/colors.h>
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Color_RGB(unsigned char r, unsigned char g, unsigned char b)
-{
-    if (driver->Color)
-	(*driver->Color)(r, g, b);
-}
-
-void COM_Standard_color(int number)
-{
-    struct color_rgb rgb;
-
-    if (number < 0 || number >= G_num_standard_colors())
-	return;
-
-    rgb = G_standard_color_rgb(number);
-    COM_Color_RGB(rgb.r, rgb.g, rgb.b);
-}

Deleted: grass/trunk/lib/driver/Draw.c
===================================================================
--- grass/trunk/lib/driver/Draw.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Draw.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,52 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Bitmap(int ncols, int nrows, int threshold,
-		const unsigned char *buf)
-{
-    if (driver->Bitmap)
-	(*driver->Bitmap) (ncols, nrows, threshold, buf);
-}
-
-void COM_Begin(void)
-{
-    if (driver->Begin)
-	(*driver->Begin)();
-}
-
-void COM_Move(double x, double y)
-{
-    if (driver->Move)
-	(*driver->Move)(x, y);
-}
-
-void COM_Cont(double x, double y)
-{
-    if (driver->Cont)
-	(*driver->Cont)(x, y);
-}
-
-void COM_Close(void)
-{
-    if (driver->Close)
-	(*driver->Close)();
-}
-
-void COM_Stroke(void)
-{
-    if (driver->Stroke)
-	(*driver->Stroke)();
-}
-
-void COM_Fill(void)
-{
-    if (driver->Fill)
-	(*driver->Fill)();
-}
-
-void COM_Point(double x, double y)
-{
-    if (driver->Point)
-	(*driver->Point)(x, y);
-}
-

Deleted: grass/trunk/lib/driver/Erase.c
===================================================================
--- grass/trunk/lib/driver/Erase.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Erase.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,8 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Erase(void)
-{
-    if (driver->Erase)
-	(*driver->Erase)();
-}

Deleted: grass/trunk/lib/driver/Font.c
===================================================================
--- grass/trunk/lib/driver/Font.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Font.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,159 +0,0 @@
-#include <stdio.h>
-#include <string.h>
-
-#include <grass/gis.h>
-#include "driver.h"
-#include "driverlib.h"
-
-static int font_type = GFONT_STROKE;
-static char *encoding;
-
-static void stroke_set(const char *filename)
-{
-    if (font_init(filename) == 0)
-	font_type = GFONT_STROKE;
-}
-
-static void freetype_set(const char *filename, int index)
-{
-    if (font_init_freetype(filename, index) == 0)
-	font_type = GFONT_FREETYPE;
-}
-
-static void driver_set(const char *name)
-{
-    (*driver->Set_font)(name);
-    font_type = GFONT_DRIVER;
-}
-
-int font_get_type(void)
-{
-    return font_type;
-}
-
-const char *font_get_encoding(void)
-{
-    if (!encoding)
-	encoding = G_store("ISO-8859-1");
-    return encoding;
-}
-
-static void font_list(char ***list, int *count, int verbose)
-{
-    char **fonts;
-    int num_fonts;
-    int i;
-
-    for (i = 0; ftcap[i].name; i++)
-	;
-    num_fonts = i;
-
-    fonts = G_malloc(num_fonts * sizeof(const char *));
-
-    for (i = 0; i < num_fonts; i++) {
-	struct GFONT_CAP *p = &ftcap[i];
-
-	if (verbose) {
-	    char buf[GPATH_MAX];
-
-	    sprintf(buf, "%s|%s|%d|%s|%d|%s|",
-		    p->name, p->longname, p->type,
-		    p->path, p->index, p->encoding);
-
-	    fonts[i] = G_store(buf);
-	}
-	else
-	    fonts[i] = G_store(p->name);
-    }
-
-    *list = fonts;
-    *count = num_fonts;
-}
-
-static void free_font_list(char **fonts, int count)
-{
-    int i;
-
-    for (i = 0; i < count; i++)
-	G_free(fonts[i]);
-
-    G_free(fonts);
-}
-
-void COM_Set_font(const char *name)
-{
-    int i;
-
-    if (G_is_absolute_path(name)) {
-	if (font_exists(name))
-	    freetype_set(name, 0);
-	return;
-    }
-
-    for (i = 0; ftcap[i].name; i++) {
-	struct GFONT_CAP *cap = &ftcap[i];
-
-	if (strcmp(name, cap->name) != 0)
-	    continue;
-
-	switch (cap->type) {
-	case GFONT_FREETYPE:
-	    freetype_set(cap->path, cap->index);
-	    COM_Set_encoding(cap->encoding);
-	    break;
-	case GFONT_STROKE:
-	    stroke_set(cap->name);
-	    break;
-	}
-	return;
-    }
-
-
-    if (driver->Font_info && driver->Set_font) {
-	char **list = NULL;
-	int count = 0;
-
-	(*driver->Font_info)(&list, &count);
-
-	for (i = 0; i < count; i++) {
-	    struct GFONT_CAP cap;
-
-	    if (!parse_fontcap_entry(&cap, list[i]))
-		continue;
-
-	    if (cap.type != GFONT_DRIVER || strcmp(name, cap.name) != 0)
-		continue;
-
-	    driver_set(cap.name);
-	    COM_Set_encoding(cap.encoding);
-	    break;
-	}
-
-	free_font_list(list, count);
-	return;
-    }
-
-    stroke_set("romans");
-}
-
-void COM_Set_encoding(const char *enc)
-{
-    if (encoding)
-	G_free(encoding);
-
-    encoding = G_store(enc);
-}
-
-void COM_Font_list(char ***list, int *count)
-{
-    font_list(list, count, 0);
-    if (driver->Font_list)
-	(*driver->Font_list)(list, count);
-}
-
-void COM_Font_info(char ***list, int *count)
-{
-    font_list(list, count, 1);
-    if (driver->Font_info)
-	(*driver->Font_info)(list, count);
-}

Deleted: grass/trunk/lib/driver/Get_t_box.c
===================================================================
--- grass/trunk/lib/driver/Get_t_box.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Get_t_box.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,19 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Get_text_box(const char *text, double *t, double *b, double *l, double *r)
-{
-    switch (font_get_type()) {
-    case GFONT_STROKE:
-	get_text_ext(text, t, b, l, r);
-	break;
-    case GFONT_FREETYPE:
-	get_text_ext_freetype(text, t, b, l, r);
-	break;
-    case GFONT_DRIVER:
-	if (driver->Text_box)
-	    (*driver->Text_box)(text, t, b, l, r);
-	break;
-    }
-}
-

Deleted: grass/trunk/lib/driver/Graph.c
===================================================================
--- grass/trunk/lib/driver/Graph.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Graph.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,15 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-int COM_Graph_set(void)
-{
-    if (driver->Graph_set)
-	return (*driver->Graph_set) ();
-    return 0;
-}
-
-void COM_Graph_close(void)
-{
-    if (driver->Graph_close)
-	(*driver->Graph_close) ();
-}

Deleted: grass/trunk/lib/driver/Line_width.c
===================================================================
--- grass/trunk/lib/driver/Line_width.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Line_width.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,8 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Line_width(double width)
-{
-    if (driver->Line_width)
-	(*driver->Line_width) (width);
-}

Deleted: grass/trunk/lib/driver/Move.c
===================================================================
--- grass/trunk/lib/driver/Move.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Move.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,9 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Pos_abs(double x, double y)
-{
-    cur_x = x;
-    cur_y = y;
-}
-

Deleted: grass/trunk/lib/driver/Raster.c
===================================================================
--- grass/trunk/lib/driver/Raster.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Raster.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,33 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <grass/gis.h>
-#include "driver.h"
-#include "driverlib.h"
-
-/******************************************************************************
- * These routines support the drawing of multi-band images on the graphics
- * device.
- ******************************************************************************
- */
-
-void COM_begin_raster(int mask, int src[2][2], double dst[2][2])
-{
-    if (driver->Begin_raster)
-	(*driver->Begin_raster) (mask, src, dst);
-}
-
-int COM_raster(int n, int row,
-	       const unsigned char *red, const unsigned char *grn,
-	       const unsigned char *blu, const unsigned char *nul)
-{
-    if (driver->Raster)
-	return (*driver->Raster) (n, row, red, grn, blu, nul);
-
-    return -1;
-}
-
-void COM_end_raster(void)
-{
-    if (driver->End_raster)
-	(*driver->End_raster) ();
-}

Deleted: grass/trunk/lib/driver/Set_window.c
===================================================================
--- grass/trunk/lib/driver/Set_window.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Set_window.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,25 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-static struct {
-    double t, b, l, r;
-} window;
-
-void COM_Set_window(double t, double b, double l, double r)
-{
-    window.t = t;
-    window.b = b;
-    window.l = l;
-    window.r = r;
-
-    if (driver->Set_window)
-	(*driver->Set_window) (t, b, l, r);
-}
-
-void COM_Get_window(double *t, double *b, double *l, double *r)
-{
-    *t = window.t;
-    *b = window.b;
-    *l = window.l;
-    *r = window.r;
-}

Deleted: grass/trunk/lib/driver/Text.c
===================================================================
--- grass/trunk/lib/driver/Text.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Text.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,19 +0,0 @@
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Text(const char *text)
-{
-    switch (font_get_type()) {
-    case GFONT_STROKE:
-	soft_text(text);
-	break;
-    case GFONT_FREETYPE:
-	soft_text_freetype(text);
-	break;
-    case GFONT_DRIVER:
-	if (driver->Text)
-	    (*driver->Text)(text);
-	break;
-    }
-}
-

Deleted: grass/trunk/lib/driver/Text_size.c
===================================================================
--- grass/trunk/lib/driver/Text_size.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/driver/Text_size.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,18 +0,0 @@
-#include <math.h>
-#include "driver.h"
-#include "driverlib.h"
-
-void COM_Text_size(double x, double y)
-{
-    text_size_x = x;
-    text_size_y = y;
-    matrix_valid = 0;
-}
-
-void COM_Text_rotation(double val)
-{
-    text_rotation = val;
-    text_sinrot = sin(M_PI * text_rotation / 180.0);
-    text_cosrot = cos(M_PI * text_rotation / 180.0);
-    matrix_valid = 0;
-}

Copied: grass/trunk/lib/driver/box.c (from rev 62429, grass/trunk/lib/driver/Box.c)
===================================================================
--- grass/trunk/lib/driver/box.c	                        (rev 0)
+++ grass/trunk/lib/driver/box.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,9 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Box_abs(double x1, double y1, double x2, double y2)
+{
+    if (driver->Box)
+	(*driver->Box)(x1, y1, x2, y2);
+}
+

Copied: grass/trunk/lib/driver/color.c (from rev 62429, grass/trunk/lib/driver/Color.c)
===================================================================
--- grass/trunk/lib/driver/color.c	                        (rev 0)
+++ grass/trunk/lib/driver/color.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,20 @@
+#include <grass/colors.h>
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Color_RGB(unsigned char r, unsigned char g, unsigned char b)
+{
+    if (driver->Color)
+	(*driver->Color)(r, g, b);
+}
+
+void COM_Standard_color(int number)
+{
+    struct color_rgb rgb;
+
+    if (number < 0 || number >= G_num_standard_colors())
+	return;
+
+    rgb = G_standard_color_rgb(number);
+    COM_Color_RGB(rgb.r, rgb.g, rgb.b);
+}

Copied: grass/trunk/lib/driver/draw.c (from rev 62429, grass/trunk/lib/driver/Draw.c)
===================================================================
--- grass/trunk/lib/driver/draw.c	                        (rev 0)
+++ grass/trunk/lib/driver/draw.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,52 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Bitmap(int ncols, int nrows, int threshold,
+		const unsigned char *buf)
+{
+    if (driver->Bitmap)
+	(*driver->Bitmap) (ncols, nrows, threshold, buf);
+}
+
+void COM_Begin(void)
+{
+    if (driver->Begin)
+	(*driver->Begin)();
+}
+
+void COM_Move(double x, double y)
+{
+    if (driver->Move)
+	(*driver->Move)(x, y);
+}
+
+void COM_Cont(double x, double y)
+{
+    if (driver->Cont)
+	(*driver->Cont)(x, y);
+}
+
+void COM_Close(void)
+{
+    if (driver->Close)
+	(*driver->Close)();
+}
+
+void COM_Stroke(void)
+{
+    if (driver->Stroke)
+	(*driver->Stroke)();
+}
+
+void COM_Fill(void)
+{
+    if (driver->Fill)
+	(*driver->Fill)();
+}
+
+void COM_Point(double x, double y)
+{
+    if (driver->Point)
+	(*driver->Point)(x, y);
+}
+

Copied: grass/trunk/lib/driver/erase.c (from rev 62429, grass/trunk/lib/driver/Erase.c)
===================================================================
--- grass/trunk/lib/driver/erase.c	                        (rev 0)
+++ grass/trunk/lib/driver/erase.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,8 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Erase(void)
+{
+    if (driver->Erase)
+	(*driver->Erase)();
+}

Copied: grass/trunk/lib/driver/font.c (from rev 62429, grass/trunk/lib/driver/Font.c)
===================================================================
--- grass/trunk/lib/driver/font.c	                        (rev 0)
+++ grass/trunk/lib/driver/font.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,159 @@
+#include <stdio.h>
+#include <string.h>
+
+#include <grass/gis.h>
+#include "driver.h"
+#include "driverlib.h"
+
+static int font_type = GFONT_STROKE;
+static char *encoding;
+
+static void stroke_set(const char *filename)
+{
+    if (font_init(filename) == 0)
+	font_type = GFONT_STROKE;
+}
+
+static void freetype_set(const char *filename, int index)
+{
+    if (font_init_freetype(filename, index) == 0)
+	font_type = GFONT_FREETYPE;
+}
+
+static void driver_set(const char *name)
+{
+    (*driver->Set_font)(name);
+    font_type = GFONT_DRIVER;
+}
+
+int font_get_type(void)
+{
+    return font_type;
+}
+
+const char *font_get_encoding(void)
+{
+    if (!encoding)
+	encoding = G_store("ISO-8859-1");
+    return encoding;
+}
+
+static void font_list(char ***list, int *count, int verbose)
+{
+    char **fonts;
+    int num_fonts;
+    int i;
+
+    for (i = 0; ftcap[i].name; i++)
+	;
+    num_fonts = i;
+
+    fonts = G_malloc(num_fonts * sizeof(const char *));
+
+    for (i = 0; i < num_fonts; i++) {
+	struct GFONT_CAP *p = &ftcap[i];
+
+	if (verbose) {
+	    char buf[GPATH_MAX];
+
+	    sprintf(buf, "%s|%s|%d|%s|%d|%s|",
+		    p->name, p->longname, p->type,
+		    p->path, p->index, p->encoding);
+
+	    fonts[i] = G_store(buf);
+	}
+	else
+	    fonts[i] = G_store(p->name);
+    }
+
+    *list = fonts;
+    *count = num_fonts;
+}
+
+static void free_font_list(char **fonts, int count)
+{
+    int i;
+
+    for (i = 0; i < count; i++)
+	G_free(fonts[i]);
+
+    G_free(fonts);
+}
+
+void COM_Set_font(const char *name)
+{
+    int i;
+
+    if (G_is_absolute_path(name)) {
+	if (font_exists(name))
+	    freetype_set(name, 0);
+	return;
+    }
+
+    for (i = 0; ftcap[i].name; i++) {
+	struct GFONT_CAP *cap = &ftcap[i];
+
+	if (strcmp(name, cap->name) != 0)
+	    continue;
+
+	switch (cap->type) {
+	case GFONT_FREETYPE:
+	    freetype_set(cap->path, cap->index);
+	    COM_Set_encoding(cap->encoding);
+	    break;
+	case GFONT_STROKE:
+	    stroke_set(cap->name);
+	    break;
+	}
+	return;
+    }
+
+
+    if (driver->Font_info && driver->Set_font) {
+	char **list = NULL;
+	int count = 0;
+
+	(*driver->Font_info)(&list, &count);
+
+	for (i = 0; i < count; i++) {
+	    struct GFONT_CAP cap;
+
+	    if (!parse_fontcap_entry(&cap, list[i]))
+		continue;
+
+	    if (cap.type != GFONT_DRIVER || strcmp(name, cap.name) != 0)
+		continue;
+
+	    driver_set(cap.name);
+	    COM_Set_encoding(cap.encoding);
+	    break;
+	}
+
+	free_font_list(list, count);
+	return;
+    }
+
+    stroke_set("romans");
+}
+
+void COM_Set_encoding(const char *enc)
+{
+    if (encoding)
+	G_free(encoding);
+
+    encoding = G_store(enc);
+}
+
+void COM_Font_list(char ***list, int *count)
+{
+    font_list(list, count, 0);
+    if (driver->Font_list)
+	(*driver->Font_list)(list, count);
+}
+
+void COM_Font_info(char ***list, int *count)
+{
+    font_list(list, count, 1);
+    if (driver->Font_info)
+	(*driver->Font_info)(list, count);
+}

Copied: grass/trunk/lib/driver/get_t_box.c (from rev 62429, grass/trunk/lib/driver/Get_t_box.c)
===================================================================
--- grass/trunk/lib/driver/get_t_box.c	                        (rev 0)
+++ grass/trunk/lib/driver/get_t_box.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,19 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Get_text_box(const char *text, double *t, double *b, double *l, double *r)
+{
+    switch (font_get_type()) {
+    case GFONT_STROKE:
+	get_text_ext(text, t, b, l, r);
+	break;
+    case GFONT_FREETYPE:
+	get_text_ext_freetype(text, t, b, l, r);
+	break;
+    case GFONT_DRIVER:
+	if (driver->Text_box)
+	    (*driver->Text_box)(text, t, b, l, r);
+	break;
+    }
+}
+

Copied: grass/trunk/lib/driver/graph.c (from rev 62429, grass/trunk/lib/driver/Graph.c)
===================================================================
--- grass/trunk/lib/driver/graph.c	                        (rev 0)
+++ grass/trunk/lib/driver/graph.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,15 @@
+#include "driver.h"
+#include "driverlib.h"
+
+int COM_Graph_set(void)
+{
+    if (driver->Graph_set)
+	return (*driver->Graph_set) ();
+    return 0;
+}
+
+void COM_Graph_close(void)
+{
+    if (driver->Graph_close)
+	(*driver->Graph_close) ();
+}

Copied: grass/trunk/lib/driver/line_width.c (from rev 62429, grass/trunk/lib/driver/Line_width.c)
===================================================================
--- grass/trunk/lib/driver/line_width.c	                        (rev 0)
+++ grass/trunk/lib/driver/line_width.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,8 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Line_width(double width)
+{
+    if (driver->Line_width)
+	(*driver->Line_width) (width);
+}

Copied: grass/trunk/lib/driver/move.c (from rev 62429, grass/trunk/lib/driver/Move.c)
===================================================================
--- grass/trunk/lib/driver/move.c	                        (rev 0)
+++ grass/trunk/lib/driver/move.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,9 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Pos_abs(double x, double y)
+{
+    cur_x = x;
+    cur_y = y;
+}
+

Copied: grass/trunk/lib/driver/raster.c (from rev 62429, grass/trunk/lib/driver/Raster.c)
===================================================================
--- grass/trunk/lib/driver/raster.c	                        (rev 0)
+++ grass/trunk/lib/driver/raster.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,33 @@
+#include <stdio.h>
+#include <stdlib.h>
+#include <grass/gis.h>
+#include "driver.h"
+#include "driverlib.h"
+
+/******************************************************************************
+ * These routines support the drawing of multi-band images on the graphics
+ * device.
+ ******************************************************************************
+ */
+
+void COM_begin_raster(int mask, int src[2][2], double dst[2][2])
+{
+    if (driver->Begin_raster)
+	(*driver->Begin_raster) (mask, src, dst);
+}
+
+int COM_raster(int n, int row,
+	       const unsigned char *red, const unsigned char *grn,
+	       const unsigned char *blu, const unsigned char *nul)
+{
+    if (driver->Raster)
+	return (*driver->Raster) (n, row, red, grn, blu, nul);
+
+    return -1;
+}
+
+void COM_end_raster(void)
+{
+    if (driver->End_raster)
+	(*driver->End_raster) ();
+}

Copied: grass/trunk/lib/driver/set_window.c (from rev 62429, grass/trunk/lib/driver/Set_window.c)
===================================================================
--- grass/trunk/lib/driver/set_window.c	                        (rev 0)
+++ grass/trunk/lib/driver/set_window.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,25 @@
+#include "driver.h"
+#include "driverlib.h"
+
+static struct {
+    double t, b, l, r;
+} window;
+
+void COM_Set_window(double t, double b, double l, double r)
+{
+    window.t = t;
+    window.b = b;
+    window.l = l;
+    window.r = r;
+
+    if (driver->Set_window)
+	(*driver->Set_window) (t, b, l, r);
+}
+
+void COM_Get_window(double *t, double *b, double *l, double *r)
+{
+    *t = window.t;
+    *b = window.b;
+    *l = window.l;
+    *r = window.r;
+}

Copied: grass/trunk/lib/driver/text.c (from rev 62429, grass/trunk/lib/driver/Text.c)
===================================================================
--- grass/trunk/lib/driver/text.c	                        (rev 0)
+++ grass/trunk/lib/driver/text.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,19 @@
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Text(const char *text)
+{
+    switch (font_get_type()) {
+    case GFONT_STROKE:
+	soft_text(text);
+	break;
+    case GFONT_FREETYPE:
+	soft_text_freetype(text);
+	break;
+    case GFONT_DRIVER:
+	if (driver->Text)
+	    (*driver->Text)(text);
+	break;
+    }
+}
+

Copied: grass/trunk/lib/driver/text_size.c (from rev 62429, grass/trunk/lib/driver/Text_size.c)
===================================================================
--- grass/trunk/lib/driver/text_size.c	                        (rev 0)
+++ grass/trunk/lib/driver/text_size.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,18 @@
+#include <math.h>
+#include "driver.h"
+#include "driverlib.h"
+
+void COM_Text_size(double x, double y)
+{
+    text_size_x = x;
+    text_size_y = y;
+    matrix_valid = 0;
+}
+
+void COM_Text_rotation(double val)
+{
+    text_rotation = val;
+    text_sinrot = sin(M_PI * text_rotation / 180.0);
+    text_cosrot = cos(M_PI * text_rotation / 180.0);
+    matrix_valid = 0;
+}

Deleted: grass/trunk/lib/gpde/N_arrays.c
===================================================================
--- grass/trunk/lib/gpde/N_arrays.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_arrays.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,1244 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*
-* PURPOSE:     	Array managment functions
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <math.h>
-
-#include <grass/N_pde.h>
-#include <grass/raster.h>
-#include <grass/glocale.h>
-
-
-/* ******************** 2D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Allocate memory for a N_array_2d data structure.
- *
- * This function allocates memory for an array of type N_array_2d
- * and returns the pointer to the new allocated memory.
- * <br><br>
- * The data type of this array is set by "type" and must be
- * CELL_TYPE, FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
- * The offset sets the number of boundary cols and rows.
- * This option is useful to generate homogeneous Neumann boundary conditions around
- * an array or to establish overlapping boundaries. The array is initialized with 0 by default.
- * <br><br>
- * If the offset is greater then 0, negative indices are possible.
- * <br><br>
- *
- * The data structure of a array with 3 rows and cols and an offset of 1
- * will looks like this:
- * <br><br>
- *
- \verbatim
- 0 0 0 0 0
- 0 0 1 2 0
- 0 3 4 5 0
- 0 6 7 8 0
- 0 0 0 0 0
- \endverbatim
- *
- * 0 is the boundary.
- * <br><br>
- * Internal a one dimensional array is allocated to save memory and to speed up the memory access.
- * To access the one dimensional array with a two dimensional index use the provided
- * get and put functions. The internal representation of the above data will look like this:
- *
- \verbatim
- 0 0 0 0 0 0 0 1 2 0 0 3 4 5 0 0 6 7 8 0 0 0 0 0 0
- \endverbatim
- *
- * \param cols int
- * \param rows int
- * \param offset int
- * \param type int
- * \return N_array_2d *
- *
- * */
-N_array_2d *N_alloc_array_2d(int cols, int rows, int offset, int type)
-{
-    N_array_2d *data = NULL;
-
-    if (rows < 1 || cols < 1)
-	G_fatal_error("N_alloc_array_2d: cols and rows should be > 0");
-
-    if (type != CELL_TYPE && type != FCELL_TYPE && type != DCELL_TYPE)
-	G_fatal_error
-	    ("N_alloc_array_2d: Wrong data type, should be CELL_TYPE, FCELL_TYPE or DCELL_TYPE");
-
-    data = (N_array_2d *) G_calloc(1, sizeof(N_array_2d));
-
-    data->cols = cols;
-    data->rows = rows;
-    data->type = type;
-    data->offset = offset;
-    data->rows_intern = rows + 2 * offset;	/*offset position at booth sides */
-    data->cols_intern = cols + 2 * offset;	/*offset position at booth sides */
-    data->cell_array = NULL;
-    data->fcell_array = NULL;
-    data->dcell_array = NULL;
-
-    if (data->type == CELL_TYPE) {
-	data->cell_array =
-	    (CELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
-			      sizeof(CELL));
-	G_debug(3,
-		"N_alloc_array_2d: CELL array allocated rows_intern %i cols_intern %i offset %i",
-		data->rows_intern, data->cols_intern, data->offset = offset);
-    }
-    else if (data->type == FCELL_TYPE) {
-	data->fcell_array =
-	    (FCELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
-			       sizeof(FCELL));
-	G_debug(3,
-		"N_alloc_array_2d: FCELL array allocated rows_intern %i cols_intern %i offset %i",
-		data->rows_intern, data->cols_intern, data->offset = offset);
-
-    }
-    else if (data->type == DCELL_TYPE) {
-	data->dcell_array =
-	    (DCELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
-			       sizeof(DCELL));
-	G_debug(3,
-		"N_alloc_array_2d: DCELL array allocated rows_intern %i cols_intern %i offset %i",
-		data->rows_intern, data->cols_intern, data->offset = offset);
-    }
-
-    return data;
-}
-
-/*!
- * \brief Release the memory of a N_array_2d structure
- *
- * \param data N_array_2d *
- * \return void
- * */
-void N_free_array_2d(N_array_2d * data)
-{
-
-    if (data != NULL) {
-	G_debug(3, "N_free_array_2d: free N_array_2d");
-
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    G_free(data->cell_array);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_free(data->fcell_array);
-
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_free(data->dcell_array);
-	}
-
-	G_free(data);
-	data = NULL;
-
-    }
-
-    return;
-}
-
-
-/*!
- * \brief Return the data type of the N_array_2d struct
- *
- * The data type can be CELL_TYPE, FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
- *
- * \param array N_array_2d *
- * \return type int
- * */
-int N_get_array_2d_type(N_array_2d * array)
-{
-    return array->type;
-}
-
-/*!
- * \brief Write the value of the N_array_2d struct at position col, row to value
- *
- * The value must be of the same type as the array. Otherwise you will risk data losses.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \param value void * - this variable contains the array value at col, row position
- * \return void
- * */
-
-void N_get_array_2d_value(N_array_2d * data, int col, int row, void *value)
-{
-
-    if (data->offset == 0) {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    *((CELL *) value) =
-		data->cell_array[row * data->cols_intern + col];
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    *((FCELL *) value) =
-		data->fcell_array[row * data->cols_intern + col];
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    *((DCELL *) value) =
-		data->dcell_array[row * data->cols_intern + col];
-	}
-    }
-    else {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    *((CELL *) value) =
-		data->cell_array[(row + data->offset) * data->cols_intern +
-				 col + data->offset];
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    *((FCELL *) value) =
-		data->fcell_array[(row + data->offset) * data->cols_intern +
-				  col + data->offset];
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    *((DCELL *) value) =
-		data->dcell_array[(row + data->offset) * data->cols_intern +
-				  col + data->offset];
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Returns 1 if the value of N_array_2d struct at postion col, row
- * is of type null, otherwise 0
- *
- * This function checks automatically the type of the array and checks for the
- * data type null value.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \return int - 1 = is null, 0 otherwise
- * */
-int N_is_array_2d_value_null(N_array_2d * data, int col, int row)
-{
-
-    if (data->offset == 0) {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type CELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     cell_array[row * data->cols_intern +
-						col]), CELL_TYPE);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type FCELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     fcell_array[row * data->cols_intern +
-						 col]), FCELL_TYPE);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type DCELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     dcell_array[row * data->cols_intern +
-						 col]), DCELL_TYPE);
-	}
-    }
-    else {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type CELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     cell_array[(row +
-						 data->offset) *
-						data->cols_intern + col +
-						data->offset]), CELL_TYPE);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type FCELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     fcell_array[(row +
-						  data->offset) *
-						 data->cols_intern + col +
-						 data->offset]), FCELL_TYPE);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_2d_value_null: null value is of type DCELL at pos [%i][%i]",
-		    col, row);
-	    return Rast_is_null_value((void *)
-				   &(data->
-				     dcell_array[(row +
-						  data->offset) *
-						 data->cols_intern + col +
-						 data->offset]), DCELL_TYPE);
-	}
-    }
-
-    return 0;
-}
-
-
-/*!
- * \brief Returns the value of type CELL at position col, row
- *
- * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the CELL type.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \return CELL
- *
- * */
-CELL N_get_array_2d_c_value(N_array_2d * data, int col, int row)
-{
-    CELL value = 0;
-    FCELL fvalue = 0.0;
-    DCELL dvalue = 0.0;
-
-    switch (data->type) {
-    case CELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&value);
-	return (CELL) value;
-    case FCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&fvalue);
-	return (CELL) fvalue;
-    case DCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&dvalue);
-	return (CELL) dvalue;
-    }
-
-    return value;
-}
-
-/*!
- * \brief Returns the value of type FCELL at position col, row
- *
- * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the FCELL type.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \return FCELL
-
- * */
-FCELL N_get_array_2d_f_value(N_array_2d * data, int col, int row)
-{
-    CELL value = 0;
-    FCELL fvalue = 0.0;
-    DCELL dvalue = 0.0;
-
-    switch (data->type) {
-    case CELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&value);
-	return (FCELL) value;
-    case FCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&fvalue);
-	return (FCELL) fvalue;
-    case DCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&dvalue);
-	return (FCELL) dvalue;
-    }
-
-    return fvalue;
-}
-
-/*!
- * \brief Returns the value of type DCELL at position col, row
- *
- * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the DCELL type.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \return DCELL
- *
- * */
-DCELL N_get_array_2d_d_value(N_array_2d * data, int col, int row)
-{
-    CELL value = 0;
-    FCELL fvalue = 0.0;
-    DCELL dvalue = 0.0;
-
-    switch (data->type) {
-    case CELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&value);
-	return (DCELL) value;
-    case FCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&fvalue);
-	return (DCELL) fvalue;
-    case DCELL_TYPE:
-	N_get_array_2d_value(data, col, row, (void *)&dvalue);
-	return (DCELL) dvalue;
-    }
-
-    return dvalue;
-
-}
-
-/*!
- * \brief Writes a value to the N_array_2d struct at position col, row
- *
- * The value will be automatically cast to the array type.
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \param value char *
- * \return void
- * */
-void N_put_array_2d_value(N_array_2d * data, int col, int row, char *value)
-{
-
-    G_debug(6, "N_put_array_2d_value: put value to array");
-
-    if (data->offset == 0) {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    data->cell_array[row * data->cols_intern + col] =
-		*((CELL *) value);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    data->fcell_array[row * data->cols_intern + col] =
-		*((FCELL *) value);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    data->dcell_array[row * data->cols_intern + col] =
-		*((DCELL *) value);
-	}
-    }
-    else {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    data->cell_array[(row + data->offset) * data->cols_intern + col +
-			     data->offset] = *((CELL *) value);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    data->fcell_array[(row + data->offset) * data->cols_intern + col +
-			      data->offset] = *((FCELL *) value);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    data->dcell_array[(row + data->offset) * data->cols_intern + col +
-			      data->offset] = *((DCELL *) value);
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Writes the null value to the N_array_2d struct at position col, row
- *
- * The null value will be automatically set to the array data type (CELL, FCELL or DCELL).
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \return void
- * */
-void N_put_array_2d_value_null(N_array_2d * data, int col, int row)
-{
-
-    G_debug(6,
-	    "N_put_array_2d_value_null: put null value to array pos [%i][%i]",
-	    col, row);
-
-    if (data->offset == 0) {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    Rast_set_c_null_value((void *)
-			       &(data->
-				 cell_array[row * data->cols_intern + col]),
-			       1);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    Rast_set_f_null_value((void *)
-			       &(data->
-				 fcell_array[row * data->cols_intern + col]),
-			       1);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    Rast_set_d_null_value((void *)
-			       &(data->
-				 dcell_array[row * data->cols_intern + col]),
-			       1);
-	}
-    }
-    else {
-	if (data->type == CELL_TYPE && data->cell_array != NULL) {
-	    Rast_set_c_null_value((void *)
-			       &(data->
-				 cell_array[(row +
-					     data->offset) *
-					    data->cols_intern + col +
-					    data->offset]), 1);
-	}
-	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    Rast_set_f_null_value((void *)
-			       &(data->
-				 fcell_array[(row +
-					      data->offset) *
-					     data->cols_intern + col +
-					     data->offset]), 1);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    Rast_set_d_null_value((void *)
-			       &(data->
-				 dcell_array[(row +
-					      data->offset) *
-					     data->cols_intern + col +
-					     data->offset]), 1);
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Writes a CELL value to the N_array_2d struct at position col, row
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \param value CELL
- * \return void
- * */
-void N_put_array_2d_c_value(N_array_2d * data, int col, int row, CELL value)
-{
-    FCELL fvalue;
-    DCELL dvalue;
-
-    switch (data->type) {
-    case FCELL_TYPE:
-	fvalue = (FCELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&fvalue);
-	return;
-    case DCELL_TYPE:
-	dvalue = (DCELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&dvalue);
-	return;
-    }
-
-    N_put_array_2d_value(data, col, row, (char *)&value);
-
-    return;
-}
-
-/*!
- * \brief Writes a FCELL value to the N_array_2d struct at position col, row
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \param value FCELL
- * \return void
- * */
-void N_put_array_2d_f_value(N_array_2d * data, int col, int row, FCELL value)
-{
-    CELL cvalue;
-    DCELL dvalue;
-
-    switch (data->type) {
-    case CELL_TYPE:
-	cvalue = (CELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&cvalue);
-	return;
-    case DCELL_TYPE:
-	dvalue = (DCELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&dvalue);
-	return;
-    }
-
-    N_put_array_2d_value(data, col, row, (char *)&value);
-
-    return;
-}
-
-/*!
- * \brief Writes a DCELL value to the N_array_2d struct at position col, row
- *
- * \param data N_array_2d *
- * \param col int
- * \param row int
- * \param value DCELL
- * \return void
- * */
-void N_put_array_2d_d_value(N_array_2d * data, int col, int row, DCELL value)
-{
-    CELL cvalue;
-    FCELL fvalue;
-
-    switch (data->type) {
-    case CELL_TYPE:
-	cvalue = (CELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&cvalue);
-	return;
-    case FCELL_TYPE:
-	fvalue = (FCELL) value;
-	N_put_array_2d_value(data, col, row, (char *)&fvalue);
-	return;
-    }
-
-    N_put_array_2d_value(data, col, row, (char *)&value);
-
-    return;
-}
-
-/*!
- * \brief This function writes the data info of the array data to stdout
- *
- * \param data N_array_2d *
- * \return void
- * */
-void N_print_array_2d_info(N_array_2d * data)
-{
-
-    fprintf(stdout, "N_array_2d \n");
-    fprintf(stdout, "Cols %i\n", data->cols);
-    fprintf(stdout, "Rows: %i\n", data->rows);
-    fprintf(stdout, "Array type: %i\n", data->type);
-    fprintf(stdout, "Offset: %i\n", data->offset);
-    fprintf(stdout, "Internal cols: %i\n", data->cols_intern);
-    fprintf(stdout, "Internal rows: %i\n", data->rows_intern);
-    fprintf(stdout, "CELL array pointer: %p\n", data->cell_array);
-    fprintf(stdout, "FCELL array pointer: %p\n", data->fcell_array);
-    fprintf(stdout, "DCELL array pointer: %p\n", data->dcell_array);
-
-
-    return;
-}
-
-/*!
- * \brief Write info and content of the N_array_2d struct to stdout
- *
- * Offsets are ignored
- *
- * \param data N_array_2d *
- * \return void
- * */
-void N_print_array_2d(N_array_2d * data)
-{
-    int i, j;
-
-    N_print_array_2d_info(data);
-
-    for (j = 0 - data->offset; j < data->rows + data->offset; j++) {
-	for (i = 0 - data->offset; i < data->cols + data->offset; i++) {
-	    if (data->type == CELL_TYPE)
-		fprintf(stdout, "%6d ", N_get_array_2d_c_value(data, i, j));
-	    else if (data->type == FCELL_TYPE)
-		fprintf(stdout, "%6.6f ", N_get_array_2d_f_value(data, i, j));
-	    else if (data->type == DCELL_TYPE)
-		printf("%6.6f ", N_get_array_2d_d_value(data, i, j));
-	}
-	fprintf(stdout, "\n");
-    }
-    fprintf(stdout, "\n");
-
-    return;
-}
-
-
-/* ******************** 3D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Allocate memory for a N_array_3d data structure.
- *
- * This functions allocates an array of type N_array_3d and returns a pointer
- * to the new allocated memory.
- * <br><br>
- * The data type of this array set by "type" must be
- * FCELL_TYPE or DCELL_TYPE accordingly to the raster3d map data types.
- * The offsets sets the number of boundary cols, rows and depths.
- * This option is useful to generate homogeneous Neumann boundary conditions around
- * an array or to establish overlapping boundaries. The arrays are initialized with 0 by default.
- * <br><br>
- * If the offset is greater then 0, negative indices are possible.
- * The data structure of a array with 3 depths, rows and cols and an offset of 1
- * will looks like this:
- *
- \verbatim
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
-
- 0  0  0  0  0
- 0  0  1  2  0
- 0  3  4  5  0
- 0  6  7  8  0
- 0  0  0  0  0
-
- 0  0  0  0  0
- 0  9 10 11  0
- 0 12 13 14  0
- 0 15 16 17  0
- 0  0  0  0  0
-
- 0  0  0  0  0
- 0 18 19 20  0
- 0 21 22 23  0
- 0 24 25 26  0
- 0  0  0  0  0
-
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
- 0  0  0  0  0
-
- \endverbatim
-
- The depth counts from the bottom to the top.
-
- * <br><br>
- * Internal a one dimensional array is allocated to speed up the memory access.
- * To access the dimensional array with a three dimensional indexing use the provided
- * get and put functions.
- *
- * \param cols int
- * \param rows int
- * \param depths int
- * \param offset int
- * \param type int
- * \return N_array_3d *
- *
- * */
-N_array_3d *N_alloc_array_3d(int cols, int rows, int depths, int offset,
-			     int type)
-{
-    N_array_3d *data = NULL;
-
-    if (rows < 1 || cols < 1 || depths < 1)
-	G_fatal_error
-	    ("N_alloc_array_3d: depths, cols and rows should be > 0");
-
-    if (type != DCELL_TYPE && type != FCELL_TYPE)
-	G_fatal_error
-	    ("N_alloc_array_3d: Wrong data type, should be FCELL_TYPE or DCELL_TYPE");
-
-    data = (N_array_3d *) G_calloc(1, sizeof(N_array_3d));
-
-    data->cols = cols;
-    data->rows = rows;
-    data->depths = depths;
-    data->type = type;
-    data->offset = offset;
-    data->rows_intern = rows + 2 * offset;
-    data->cols_intern = cols + 2 * offset;
-    data->depths_intern = depths + 2 * offset;
-    data->fcell_array = NULL;
-    data->dcell_array = NULL;
-
-    if (data->type == FCELL_TYPE) {
-	data->fcell_array =
-	    (float *)G_calloc((size_t) data->depths_intern * data->rows_intern *
-			      data->cols_intern, sizeof(float));
-	G_debug(3,
-		"N_alloc_array_3d: float array allocated rows_intern %i cols_intern %i depths_intern %i offset %i",
-		data->rows_intern, data->cols_intern, data->depths_intern,
-		data->offset = offset);
-    }
-    else if (data->type == DCELL_TYPE) {
-	data->dcell_array =
-	    (double *)G_calloc((size_t) data->depths_intern * data->rows_intern *
-			       data->cols_intern, sizeof(double));
-	G_debug(3,
-		"N_alloc_array_3d: double array allocated rows_intern %i cols_intern %i depths_intern %i offset %i",
-		data->rows_intern, data->cols_intern, data->depths_intern,
-		data->offset = offset);
-    }
-
-    return data;
-}
-
-/*!
- * \brief Release the memory of a N_array_3d
- *
- * \param data N_array_3d *
- * \return void
- * */
-void N_free_array_3d(N_array_3d * data)
-{
-
-    if (data != NULL) {
-	G_debug(3, "N_free_array_3d: free N_array_3d");
-
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_free(data->fcell_array);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_free(data->dcell_array);
-	}
-
-	G_free(data);
-	data = NULL;
-
-    }
-
-    return;
-}
-
-/*!
- * \brief Return the data type of the N_array_3d
- *
- * The data type can be FCELL_TYPE and DCELL_TYPE accordingly to the raster map data types.
- *
- * \param array N_array_3d *
- * \return type int -- FCELL_TYPE or DCELL_TYPE
- * */
-int N_get_array_3d_type(N_array_3d * array)
-{
-    return array->type;
-}
-
-
-/*!
- * \brief This function writes the value of N_array_3d data at position col, row, depth
- *        to the variable value
- *
- * The value must be from the same type as the array. Otherwise you will risk data losses.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \param value void *
- * \return void
- * */
-void
-N_get_array_3d_value(N_array_3d * data, int col, int row, int depth,
-		     void *value)
-{
-
-    if (data->offset == 0) {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    *((float *)value) =
-		data->fcell_array[depth *
-				  (data->rows_intern * data->cols_intern) +
-				  row * data->cols_intern + col];
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    *((double *)value) =
-		data->dcell_array[depth *
-				  (data->rows_intern * data->cols_intern) +
-				  row * data->cols_intern + col];
-	}
-    }
-    else {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    *((float *)value) =
-		data->fcell_array[(depth + data->offset) *
-				  (data->rows_intern * data->cols_intern) +
-				  (row + data->offset) * data->cols_intern +
-				  (col + data->offset)];
-
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    *((double *)value) =
-		data->dcell_array[(depth + data->offset) *
-				  (data->rows_intern * data->cols_intern) +
-				  (row + data->offset) * data->cols_intern +
-				  (col + data->offset)];
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief This function returns 1 if value of N_array_3d data at position col, row, depth
- * is of type null, otherwise 0
- *
- * This function checks automatically the type of the array and checks for the
- * data type null value.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \return void
- * */
-int N_is_array_3d_value_null(N_array_3d * data, int col, int row, int depth)
-{
-
-    if (data->offset == 0) {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
-		    depth, row, col);
-	    return Rast3d_is_null_value_num((void *)
-				      &(data->
-					fcell_array[depth *
-						    (data->rows_intern *
-						     data->cols_intern) +
-						    row * data->cols_intern +
-						    col]), FCELL_TYPE);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
-		    depth, row, col);
-	    return Rast3d_is_null_value_num((void *)
-				      &(data->
-					dcell_array[depth *
-						    (data->rows_intern *
-						     data->cols_intern) +
-						    row * data->cols_intern +
-						    col]), DCELL_TYPE);
-	}
-    }
-    else {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
-		    depth, row, col);
-	    return Rast3d_is_null_value_num((void *)
-				      &(data->
-					fcell_array[(depth +
-						     data->offset) *
-						    (data->rows_intern *
-						     data->cols_intern) +
-						    (row + data->offset)
-						    * data->cols_intern +
-						    (col + data->offset)]),
-				      FCELL_TYPE);
-
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    G_debug(6,
-		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
-		    depth, row, col);
-	    return Rast3d_is_null_value_num((void *)
-				      &(data->
-					dcell_array[(depth +
-						     data->offset) *
-						    (data->rows_intern *
-						     data->cols_intern) +
-						    (row +
-						     data->offset) *
-						    data->cols_intern + (col +
-									 data->
-									 offset)]),
-				      DCELL_TYPE);
-	}
-    }
-
-    return 0;
-}
-
-/*!
- * \brief This function returns the value of type float at position col, row, depth
- *
- * The data type can be FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \return float
- *
- * */
-float N_get_array_3d_f_value(N_array_3d * data, int col, int row, int depth)
-{
-    float fvalue = 0.0;
-    double dvalue = 0.0;
-
-    switch (data->type) {
-    case FCELL_TYPE:
-	N_get_array_3d_value(data, col, row, depth, (void *)&fvalue);
-	return (float)fvalue;
-    case DCELL_TYPE:
-	N_get_array_3d_value(data, col, row, depth, (void *)&dvalue);
-	return (float)dvalue;
-    }
-
-    return fvalue;
-}
-
-/*!
- * \brief This function returns the value of type float at position col, row, depth
- *
- * The data type can be FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \return double
- *
- * */
-double N_get_array_3d_d_value(N_array_3d * data, int col, int row, int depth)
-{
-    float fvalue = 0.0;
-    double dvalue = 0.0;
-
-    switch (data->type) {
-
-    case FCELL_TYPE:
-	N_get_array_3d_value(data, col, row, depth, (void *)&fvalue);
-	return (double)fvalue;
-    case DCELL_TYPE:
-	N_get_array_3d_value(data, col, row, depth, (void *)&dvalue);
-	return (double)dvalue;
-    }
-
-    return dvalue;
-}
-
-/*!
- * \brief This function writes a value to the N_array_3d data at position col, row, depth
- *
- * The value will be automatically cast to the array type.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \param value cahr *
- * \return void
- * */
-void
-N_put_array_3d_value(N_array_3d * data, int col, int row, int depth,
-		     char *value)
-{
-
-    G_debug(6, "N_put_array_3d_value: put value to array at pos [%i][%i][%i]",
-	    depth, row, col);
-
-    if (data->offset == 0) {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    data->fcell_array[depth *
-			      (data->rows_intern * data->cols_intern) +
-			      row * data->cols_intern + col]
-		= *((float *)value);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-
-	    data->dcell_array[depth *
-			      (data->rows_intern * data->cols_intern) +
-			      row * data->cols_intern + col]
-		= *((double *)value);
-	}
-    }
-    else {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    data->fcell_array[(depth + data->offset) *
-			      (data->rows_intern * data->cols_intern) + (row +
-									 data->
-									 offset)
-			      * data->cols_intern + (col + data->offset)] =
-		*((float *)value);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    data->dcell_array[(depth + data->offset) *
-			      (data->rows_intern * data->cols_intern) + (row +
-									 data->
-									 offset)
-			      * data->cols_intern + (col + data->offset)] =
-		*((double *)value);
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief This function writes a null value to the N_array_3d data at position col, row, depth
- *
- * The null value will be automatically set to the array type.
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \return void
- * */
-void N_put_array_3d_value_null(N_array_3d * data, int col, int row, int depth)
-{
-
-    G_debug(6,
-	    "N_put_array_3d_value_null: put null value to array at pos [%i][%i][%i]",
-	    depth, row, col);
-
-    if (data->offset == 0) {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    Rast3d_set_null_value((void *)
-			     &(data->
-			       fcell_array[depth *
-					   (data->rows_intern *
-					    data->cols_intern) +
-					   row * data->cols_intern + col]), 1,
-			     FCELL_TYPE);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    Rast3d_set_null_value((void *)
-			     &(data->
-			       dcell_array[depth *
-					   (data->rows_intern *
-					    data->cols_intern) +
-					   row * data->cols_intern + col]), 1,
-			     DCELL_TYPE);
-	}
-    }
-    else {
-	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
-	    Rast3d_set_null_value((void *)
-			     &(data->
-			       fcell_array[(depth +
-					    data->offset) *
-					   (data->rows_intern *
-					    data->cols_intern) + (row +
-								  data->
-								  offset) *
-					   data->cols_intern + (col +
-								data->
-								offset)]), 1,
-			     FCELL_TYPE);
-	}
-	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
-	    Rast3d_set_null_value((void *)
-			     &(data->
-			       dcell_array[(depth +
-					    data->offset) *
-					   (data->rows_intern *
-					    data->cols_intern) + (row +
-								  data->
-								  offset) *
-					   data->cols_intern + (col +
-								data->
-								offset)]), 1,
-			     DCELL_TYPE);
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief This function writes a float value to the N_array_3d data at position col, row, depth
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \param value float
- * \return void
- * */
-void
-N_put_array_3d_f_value(N_array_3d * data, int col, int row, int depth,
-		       float value)
-{
-    double dval;
-
-    if (data->type == DCELL_TYPE) {
-	dval = (double)value;
-	N_put_array_3d_value(data, col, row, depth, (void *)&dval);
-    }
-    else {
-	N_put_array_3d_value(data, col, row, depth, (void *)&value);
-    }
-
-    return;
-}
-
-/*!
- * \brief Writes a double value to the N_array_3d struct at position col, row, depth
- *
- * \param data N_array_3d *
- * \param col int
- * \param row int
- * \param depth int
- * \param value double
- * \return void
- * */
-void
-N_put_array_3d_d_value(N_array_3d * data, int col, int row, int depth,
-		       double value)
-{
-    float fval;
-
-    if (data->type == FCELL_TYPE) {
-	fval = (double)value;
-	N_put_array_3d_value(data, col, row, depth, (void *)&fval);
-    }
-    else {
-	N_put_array_3d_value(data, col, row, depth, (void *)&value);
-    }
-
-    return;
-}
-
-/*!
- * \brief Write the info of the array to stdout
- *
- * \param data N_array_3d *
- * \return void
- * */
-void N_print_array_3d_info(N_array_3d * data)
-{
-
-    fprintf(stdout, "N_array_3d \n");
-    fprintf(stdout, "Cols %i\n", data->cols);
-    fprintf(stdout, "Rows: %i\n", data->rows);
-    fprintf(stdout, "Depths: %i\n", data->depths);
-    fprintf(stdout, "Array type: %i\n", data->type);
-    fprintf(stdout, "Offset: %i\n", data->offset);
-    fprintf(stdout, "Internal cols: %i\n", data->cols_intern);
-    fprintf(stdout, "Internal rows: %i\n", data->rows_intern);
-    fprintf(stdout, "Internal depths: %i\n", data->depths_intern);
-    fprintf(stdout, "FCELL array pointer: %p\n", data->fcell_array);
-    fprintf(stdout, "DCELL array pointer: %p\n", data->dcell_array);
-
-    return;
-}
-
-/*!
- * \brief Write info and content of the array data to stdout
- *
- * Offsets are ignored
- *
- * \param data N_array_2d *
- * \return void
- * */
-void N_print_array_3d(N_array_3d * data)
-{
-    int i, j, k;
-
-    N_print_array_3d_info(data);
-
-    for (k = 0; k < data->depths; k++) {
-	for (j = 0; j < data->rows; j++) {
-	    for (i = 0; i < data->cols; i++) {
-		if (data->type == FCELL_TYPE)
-		    printf("%6.6f ", N_get_array_3d_f_value(data, i, j, k));
-		else if (data->type == DCELL_TYPE)
-		    printf("%6.6f ", N_get_array_3d_d_value(data, i, j, k));
-	    }
-	    printf("\n");
-	}
-	printf("\n");
-    }
-    printf("\n");
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_arrays_calc.c
===================================================================
--- grass/trunk/lib/gpde/N_arrays_calc.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_arrays_calc.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,889 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:     	Higher level array managment functions 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-#include <math.h>
-
-#include <grass/N_pde.h>
-#include <grass/raster.h>
-#include <grass/glocale.h>
-
-
-/* ******************** 2D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Copy the source N_array_2d struct to the target N_array_2d struct
- *
- * The arrays must have the same size and the same offset.
- *
- * The array types can be mixed, the values are automatically casted
- * and the null values are set accordingly.
- * <br><br>
- * If you copy a cell array into a dcell array, the values are casted to dcell and 
- * the null values are converted from cell-null to dcell-null
- * <br><br>
- * This function can be called in a parallel region defined with OpenMP.
- * The copy loop is parallelize with a openmp for pragma.
- *
- * \param source N_array_2d *
- * \param target N_array_2d *
- * \return void
- * */
-void N_copy_array_2d(N_array_2d * source, N_array_2d * target)
-{
-    int i;
-    int null = 0;
-
-#pragma omp single
-    {
-	if (source->cols_intern != target->cols_intern)
-	    G_fatal_error
-		("N_copy_array_2d: the arrays are not of equal size");
-
-	if (source->rows_intern != target->rows_intern)
-	    G_fatal_error
-		("N_copy_array_2d: the arrays are not of equal size");
-
-	G_debug(3,
-		"N_copy_array_2d: copy source array to target array size %i",
-		source->cols_intern * source->rows_intern);
-    }
-
-#pragma omp for
-    for (i = 0; i < source->cols_intern * source->rows_intern; i++) {
-	null = 0;
-	if (source->type == CELL_TYPE) {
-	    if (Rast_is_c_null_value((void *)&source->cell_array[i]))
-		null = 1;
-
-	    if (target->type == CELL_TYPE) {
-		target->cell_array[i] = source->cell_array[i];
-	    }
-	    if (target->type == FCELL_TYPE) {
-		if (null)
-		    Rast_set_f_null_value((void *)&(target->fcell_array[i]), 1);
-		else
-		    target->fcell_array[i] = (FCELL) source->cell_array[i];
-	    }
-	    if (target->type == DCELL_TYPE) {
-		if (null)
-		    Rast_set_d_null_value((void *)&(target->dcell_array[i]), 1);
-		else
-		    target->dcell_array[i] = (DCELL) source->cell_array[i];
-	    }
-
-	}
-	if (source->type == FCELL_TYPE) {
-	    if (Rast_is_f_null_value((void *)&source->fcell_array[i]))
-		null = 1;
-
-	    if (target->type == CELL_TYPE) {
-		if (null)
-		    Rast_set_c_null_value((void *)&(target->cell_array[i]), 1);
-		else
-		    target->cell_array[i] = (CELL) source->fcell_array[i];
-	    }
-	    if (target->type == FCELL_TYPE) {
-		target->fcell_array[i] = source->fcell_array[i];
-	    }
-	    if (target->type == DCELL_TYPE) {
-		if (null)
-		    Rast_set_d_null_value((void *)&(target->dcell_array[i]), 1);
-		else
-		    target->dcell_array[i] = (DCELL) source->fcell_array[i];
-	    }
-	}
-	if (source->type == DCELL_TYPE) {
-	    if (Rast_is_d_null_value((void *)&source->dcell_array[i]))
-		null = 1;
-
-	    if (target->type == CELL_TYPE) {
-		if (null)
-		    Rast_set_c_null_value((void *)&(target->cell_array[i]), 1);
-		else
-		    target->cell_array[i] = (CELL) source->dcell_array[i];
-	    }
-	    if (target->type == FCELL_TYPE) {
-		if (null)
-		    Rast_set_f_null_value((void *)&(target->fcell_array[i]), 1);
-		else
-		    target->fcell_array[i] = (FCELL) source->dcell_array[i];
-	    }
-	    if (target->type == DCELL_TYPE) {
-		target->dcell_array[i] = source->dcell_array[i];
-	    }
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Calculate the norm of the two input arrays
- *
- * The norm can be of type N_MAXIMUM_NORM or N_EUKLID_NORM.
- * All arrays must have equal sizes and offsets.
- * The complete data array inclusively offsets is used for norm calucaltion.
- * Only non-null values are used to calcualte the norm.
- *
-
- * \param a N_array_2d *
- * \param b N_array_2d *
- * \param type the type of the norm -> N_MAXIMUM_NORM, N_EUKLID_NORM
- * \return double the calculated norm
- * */
-double N_norm_array_2d(N_array_2d * a, N_array_2d * b, int type)
-{
-    int i = 0;
-    double norm = 0.0, tmp = 0.0;
-    double v1 = 0.0, v2 = 0.0;
-
-    if (a->cols_intern != b->cols_intern)
-	G_fatal_error("N_norm_array_2d: the arrays are not of equal size");
-
-    if (a->rows_intern != b->rows_intern)
-	G_fatal_error("N_norm_array_2d: the arrays are not of equal size");
-
-    G_debug(3, "N_norm_array_2d: norm of a and b size %i",
-	    a->cols_intern * a->rows_intern);
-
-    for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
-	v1 = 0.0;
-	v2 = 0.0;
-
-	if (a->type == CELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(a->cell_array[i])))
-		v1 = (double)a->cell_array[i];
-	}
-	if (a->type == FCELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(a->fcell_array[i])))
-		v1 = (double)a->fcell_array[i];
-	}
-	if (a->type == DCELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(a->dcell_array[i])))
-		v1 = (double)a->dcell_array[i];
-	}
-	if (b->type == CELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(b->cell_array[i])))
-		v2 = (double)b->cell_array[i];
-	}
-	if (b->type == FCELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(b->fcell_array[i])))
-		v2 = (double)b->fcell_array[i];
-	}
-	if (b->type == DCELL_TYPE) {
-	    if (!Rast_is_f_null_value((void *)&(b->dcell_array[i])))
-		v2 = (double)b->dcell_array[i];
-	}
-
-	if (type == N_MAXIMUM_NORM) {
-	    tmp = fabs(v2 - v1);
-	    if ((tmp > norm))
-		norm = tmp;
-	}
-	if (type == N_EUKLID_NORM) {
-	    norm += fabs(v2 - v1);
-	}
-    }
-
-    return norm;
-}
-
-/*!
- * \brief Calculate basic statistics of the N_array_2d struct 
- *
- * Calculates the minimum, maximum, sum and the number of 
- * non null values. The array offset can be included in the calculation.
- *
- * \param a N_array_2d * - input array
- * \param min double* - variable to store the computed minimum
- * \param max double* - variable to store the computed maximum
- * \param sum double* - variable to store the computed sum
- * \param nonull int* - variable to store the number of non null values
- * \param withoffset - if 1 include offset values in statistic calculation, 0 otherwise 
- * \return void
- * */
-void N_calc_array_2d_stats(N_array_2d * a, double *min, double *max,
-			   double *sum, int *nonull, int withoffset)
-{
-    int i, j;
-    double val;
-
-    *sum = 0.0;
-    *nonull = 0;
-
-    if (withoffset == 1) {
-
-	*min =
-	    (double)N_get_array_2d_d_value(a, 0 - a->offset, 0 - a->offset);
-	*max =
-	    (double)N_get_array_2d_d_value(a, 0 - a->offset, 0 - a->offset);
-
-	for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
-	    for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
-		if (!N_is_array_2d_value_null(a, i, j)) {
-		    val = (double)N_get_array_2d_d_value(a, i, j);
-		    if (*min > val)
-			*min = val;
-		    if (*max < val)
-			*max = val;
-		    *sum += val;
-		    (*nonull)++;
-		}
-	    }
-	}
-    }
-    else {
-
-	*min = (double)N_get_array_2d_d_value(a, 0, 0);
-	*max = (double)N_get_array_2d_d_value(a, 0, 0);
-
-
-	for (j = 0; j < a->rows; j++) {
-	    for (i = 0; i < a->cols; i++) {
-		if (!N_is_array_2d_value_null(a, i, j)) {
-		    val = (double)N_get_array_2d_d_value(a, i, j);
-		    if (*min > val)
-			*min = val;
-		    if (*max < val)
-			*max = val;
-		    *sum += val;
-		    (*nonull)++;
-		}
-	    }
-	}
-    }
-
-    G_debug(3,
-	    "N_calc_array_2d_stats: compute array stats, min %g, max %g, sum %g, nonull %i",
-	    *min, *max, *sum, *nonull);
-    return;
-}
-
-
-/*!
- * \brief Perform calculations with two input arrays, 
- * the result is written to a third array.
- *
- * All arrays must have equal sizes and offsets.
- * The complete data array inclusively offsets is used for calucaltions.
- * Only non-null values are computed. If one array value is null, 
- * the result array value will be null too.
- * <br><br>
- * If a division with zero is detected, the resulting arrays 
- * value will set to null and not to NaN.
- * <br><br>
- * The result array is optional, if the result arrays points to NULL,
- * a new array will be allocated with the largest arrays data type
- * (CELL, FCELL or DCELL) used by the input arrays.
- * <br><br>
- * the array computations can be of the following forms:
- *
- * <ul>
- * <li>result = a + b -> N_ARRAY_SUM</li>
- * <li>result = a - b -> N_ARRAY_DIF</li>
- * <li>result = a * b -> N_ARRAY_MUL</li>
- * <li>result = a / b -> N_ARRAY_DIV</li>
- * </ul>
- *
- * \param a N_array_2d * - first input array
- * \param b N_array_2d * - second input array
- * \param result N_array_2d * - the optional result array
- * \param type  - the type of calculation
- * \return N_array_2d * - the pointer to the result array
- * */
-N_array_2d *N_math_array_2d(N_array_2d * a, N_array_2d * b,
-			    N_array_2d * result, int type)
-{
-    N_array_2d *c;
-    int i, j, setnull = 0;
-    double va = 0.0, vb = 0.0, vc = 0.0;	/*variables used for calculation */
-
-    /*Set the pointer */
-    c = result;
-
-#pragma omp single
-    {
-	/*Check the array sizes */
-	if (a->cols_intern != b->cols_intern)
-	    G_fatal_error
-		("N_math_array_2d: the arrays are not of equal size");
-	if (a->rows_intern != b->rows_intern)
-	    G_fatal_error
-		("N_math_array_2d: the arrays are not of equal size");
-	if (a->offset != b->offset)
-	    G_fatal_error
-		("N_math_array_2d: the arrays have different offsets");
-
-	G_debug(3, "N_math_array_2d: mathematical calculations, size: %i",
-		a->cols_intern * a->rows_intern);
-
-	/*if the result array is null, allocate a new one, use the 
-	 * largest data type of the input arrays*/
-	if (c == NULL) {
-	    if (a->type == DCELL_TYPE || b->type == DCELL_TYPE) {
-		c = N_alloc_array_2d(a->cols, a->rows, a->offset, DCELL_TYPE);
-		G_debug(3,
-			"N_math_array_2d: array of type DCELL_TYPE created");
-	    }
-	    else if (a->type == FCELL_TYPE || b->type == FCELL_TYPE) {
-		c = N_alloc_array_2d(a->cols, a->rows, a->offset, FCELL_TYPE);
-		G_debug(3,
-			"N_math_array_2d: array of type FCELL_TYPE created");
-	    }
-	    else {
-		c = N_alloc_array_2d(a->cols, a->rows, a->offset, CELL_TYPE);
-		G_debug(3,
-			"N_math_array_2d: array of type CELL_TYPE created");
-	    }
-	}
-	else {
-	    /*Check the array sizes */
-	    if (a->cols_intern != c->cols_intern)
-		G_fatal_error
-		    ("N_math_array_2d: the arrays are not of equal size");
-	    if (a->rows_intern != c->rows_intern)
-		G_fatal_error
-		    ("N_math_array_2d: the arrays are not of equal size");
-	    if (a->offset != c->offset)
-		G_fatal_error
-		    ("N_math_array_2d: the arrays have different offsets");
-	}
-    }
-
-#pragma omp for private(va, vb, vc, setnull)
-    for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
-	for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
-	    if (!N_is_array_2d_value_null(a, i, j) &&
-		!N_is_array_2d_value_null(b, i, j)) {
-		/*we always calulate internally with double values */
-		va = (double)N_get_array_2d_d_value(a, i, j);
-		vb = (double)N_get_array_2d_d_value(b, i, j);
-		vc = 0;
-		setnull = 0;
-
-		switch (type) {
-		case N_ARRAY_SUM:
-		    vc = va + vb;
-		    break;
-		case N_ARRAY_DIF:
-		    vc = va - vb;
-		    break;
-		case N_ARRAY_MUL:
-		    vc = va * vb;
-		    break;
-		case N_ARRAY_DIV:
-		    if (vb != 0)
-			vc = va / vb;
-		    else
-			setnull = 1;
-		    break;
-		}
-
-		if (c->type == CELL_TYPE) {
-		    if (setnull)
-			N_put_array_2d_value_null(c, i, j);
-		    else
-			N_put_array_2d_c_value(c, i, j, (CELL) vc);
-		}
-		if (c->type == FCELL_TYPE) {
-		    if (setnull)
-			N_put_array_2d_value_null(c, i, j);
-		    else
-			N_put_array_2d_f_value(c, i, j, (FCELL) vc);
-		}
-		if (c->type == DCELL_TYPE) {
-		    if (setnull)
-			N_put_array_2d_value_null(c, i, j);
-		    else
-			N_put_array_2d_d_value(c, i, j, (DCELL) vc);
-		}
-
-	    }
-	    else {
-		N_put_array_2d_value_null(c, i, j);
-	    }
-	}
-    }
-
-    return c;
-}
-
-/*!
- * \brief Convert all null values to zero values
- *
- * The complete data array inclusively offsets is used.
- * The array data types are automatically recognized.
- *
- * \param a N_array_2d *
- * \return int - number of replaced values
- * */
-int N_convert_array_2d_null_to_zero(N_array_2d * a)
-{
-    int i = 0, count = 0;
-
-    G_debug(3, "N_convert_array_2d_null_to_zero: convert array of size %i",
-	    a->cols_intern * a->rows_intern);
-
-    if (a->type == CELL_TYPE)
-	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
-	    if (Rast_is_c_null_value((void *)&(a->cell_array[i]))) {
-		a->cell_array[i] = 0;
-		count++;
-	    }
-	}
-
-    if (a->type == FCELL_TYPE)
-	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
-	    if (Rast_is_f_null_value((void *)&(a->fcell_array[i]))) {
-		a->fcell_array[i] = 0.0;
-		count++;
-	    }
-	}
-
-
-    if (a->type == DCELL_TYPE)
-	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
-	    if (Rast_is_d_null_value((void *)&(a->dcell_array[i]))) {
-		a->dcell_array[i] = 0.0;
-		count++;
-	    }
-	}
-
-
-    if (a->type == CELL_TYPE)
-	G_debug(2,
-		"N_convert_array_2d_null_to_zero: %i values of type CELL_TYPE are converted",
-		count);
-    if (a->type == FCELL_TYPE)
-	G_debug(2,
-		"N_convert_array_2d_null_to_zero: %i valuess of type FCELL_TYPE are converted",
-		count);
-    if (a->type == DCELL_TYPE)
-	G_debug(2,
-		"N_convert_array_2d_null_to_zero: %i valuess of type DCELL_TYPE are converted",
-		count);
-
-    return count;
-}
-
-/* ******************** 3D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Copy the source N_array_3d struct to the target N_array_3d struct
- *
- * The arrays must have the same size and the same offset.
- *
- * The array data types can be mixed, the values are automatically casted
- * and the null values are set accordingly.
- *
- * If you copy a float array to a double array, the values are casted to DCELL and 
- * the null values are converted from FCELL-null to DCELL-null
- *
- * \param source N_array_3d *
- * \param target N_array_3d *
- * \return void
- * */
-void N_copy_array_3d(N_array_3d * source, N_array_3d * target)
-{
-    int i;
-    int null;
-
-    if (source->cols_intern != target->cols_intern)
-	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
-
-    if (source->rows_intern != target->rows_intern)
-	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
-
-    if (source->depths_intern != target->depths_intern)
-	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
-
-
-    G_debug(3, "N_copy_array_3d: copy source array to target array size %i",
-	    source->cols_intern * source->rows_intern *
-	    source->depths_intern);
-
-    for (i = 0;
-	 i <
-	 source->cols_intern * source->rows_intern * source->depths_intern;
-	 i++) {
-	null = 0;
-	if (source->type == FCELL_TYPE) {
-	    if (Rast3d_is_null_value_num
-		((void *)&(source->fcell_array[i]), FCELL_TYPE))
-		null = 1;
-
-	    if (target->type == FCELL_TYPE) {
-		target->fcell_array[i] = source->fcell_array[i];
-	    }
-	    if (target->type == DCELL_TYPE) {
-		if (null)
-		    Rast3d_set_null_value((void *)&(target->dcell_array[i]), 1,
-				     DCELL_TYPE);
-		else
-		    target->dcell_array[i] = (double)source->fcell_array[i];
-	    }
-
-	}
-	if (source->type == DCELL_TYPE) {
-	    if (Rast3d_is_null_value_num
-		((void *)&(source->dcell_array[i]), DCELL_TYPE))
-		null = 1;
-
-	    if (target->type == FCELL_TYPE) {
-		if (null)
-		    Rast3d_set_null_value((void *)&(target->fcell_array[i]), 1,
-				     FCELL_TYPE);
-		else
-		    target->fcell_array[i] = (float)source->dcell_array[i];
-	    }
-	    if (target->type == DCELL_TYPE) {
-		target->dcell_array[i] = source->dcell_array[i];
-	    }
-	}
-    }
-
-    return;
-}
-
-
-/*!
- * \brief Calculate the norm of the two input arrays
- *
- * The norm can be of type N_MAXIMUM_NORM or N_EUKLID_NORM.
- * All arrays must have equal sizes and offsets.
- * The complete data array inclusively offsets is used for norm calucaltion.
- * Only non-null values are used to calcualte the norm.
- *
- * \param a N_array_3d *
- * \param b N_array_3d *
- * \param type the type of the norm -> N_MAXIMUM_NORM, N_EUKLID_NORM
- * \return double the calculated norm
- * */
-double N_norm_array_3d(N_array_3d * a, N_array_3d * b, int type)
-{
-    int i = 0;
-    double norm = 0.0, tmp = 0.0;
-    double v1 = 0.0, v2 = 0.0;
-
-    if (a->cols_intern != b->cols_intern)
-	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
-
-    if (a->rows_intern != b->rows_intern)
-	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
-
-    if (a->depths_intern != b->depths_intern)
-	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
-
-    G_debug(3, "N_norm_array_3d: norm of a and b size %i",
-	    a->cols_intern * a->rows_intern * a->depths_intern);
-
-    for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern; i++) {
-	v1 = 0.0;
-	v2 = 0.0;
-
-	if (a->type == FCELL_TYPE) {
-	    if (!Rast3d_is_null_value_num((void *)&(a->fcell_array[i]), FCELL_TYPE))
-		v1 = (double)a->fcell_array[i];
-	}
-	if (a->type == DCELL_TYPE) {
-	    if (!Rast3d_is_null_value_num((void *)&(a->dcell_array[i]), DCELL_TYPE))
-		v1 = (double)a->dcell_array[i];
-	}
-	if (b->type == FCELL_TYPE) {
-	    if (!Rast3d_is_null_value_num((void *)&(b->fcell_array[i]), FCELL_TYPE))
-		v2 = (double)b->fcell_array[i];
-	}
-	if (b->type == DCELL_TYPE) {
-	    if (!Rast3d_is_null_value_num((void *)&(b->dcell_array[i]), DCELL_TYPE))
-		v2 = (double)b->dcell_array[i];
-	}
-
-	if (type == N_MAXIMUM_NORM) {
-	    tmp = fabs(v2 - v1);
-	    if ((tmp > norm))
-		norm = tmp;
-	}
-	if (type == N_EUKLID_NORM) {
-	    norm += fabs(v2 - v1);
-	}
-    }
-
-    return norm;
-}
-
-/*!
- * \brief Calculate basic statistics of the N_array_3d struct
- *
- * Calculates the minimum, maximum, sum and the number of 
- * non null values. The array offset can be included in the statistical calculation.
- *
- * \param a N_array_3d * - input array
- * \param min double* - variable to store the computed minimum
- * \param max double* - variable to store the computed maximum
- * \param sum double* - variable to store the computed sum
- * \param nonull int* - variable to store the number of non null values
- * \param withoffset - if 1 include offset values in statistic calculation, 0 otherwise 
- * \return void
- * */
-void N_calc_array_3d_stats(N_array_3d * a, double *min, double *max,
-			   double *sum, int *nonull, int withoffset)
-{
-    int i, j, k;
-    double val;
-
-    *sum = 0.0;
-    *nonull = 0;
-
-    if (withoffset == 1) {
-
-	*min =
-	    (double)N_get_array_3d_d_value(a, 0 - a->offset, 0 - a->offset,
-					   0 - a->offset);
-	*max =
-	    (double)N_get_array_3d_d_value(a, 0 - a->offset, 0 - a->offset,
-					   0 - a->offset);
-
-	for (k = 0 - a->offset; k < a->depths + a->offset; k++) {
-	    for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
-		for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
-		    if (!N_is_array_3d_value_null(a, i, j, k)) {
-			val = (double)N_get_array_3d_d_value(a, i, j, k);
-			if (*min > val)
-			    *min = val;
-			if (*max < val)
-			    *max = val;
-			*sum += val;
-			(*nonull)++;
-		    }
-		}
-	    }
-	}
-    }
-    else {
-
-	*min = (double)N_get_array_3d_d_value(a, 0, 0, 0);
-	*max = (double)N_get_array_3d_d_value(a, 0, 0, 0);
-
-	for (k = 0; k < a->depths; k++) {
-	    for (j = 0; j < a->rows; j++) {
-		for (i = 0; i < a->cols; i++) {
-		    if (!N_is_array_3d_value_null(a, i, j, k)) {
-			val = (double)N_get_array_3d_d_value(a, i, j, k);
-			if (*min > val)
-			    *min = val;
-			if (*max < val)
-			    *max = val;
-			*sum += val;
-			(*nonull)++;
-		    }
-		}
-	    }
-	}
-    }
-
-    G_debug(3,
-	    "N_calc_array_3d_stats: compute array stats, min %g, max %g, sum %g, nonull %i",
-	    *min, *max, *sum, *nonull);
-
-    return;
-}
-
-/*!
- * \brief Perform calculations with two input arrays, 
- * the result is written to a third array.
- *
- * All arrays must have equal sizes and offsets.
- * The complete data array inclusively offsets is used for calucaltions.
- * Only non-null values are used. If one array value is null, 
- * the result array value will be null too.
- * <br><br>
- *
- * If a division with zero is detected, the resulting arrays 
- * value will set to null and not to NaN.
- * <br><br>
- *
- * The result array is optional, if the result arrays points to NULL,
- * a new array will be allocated with the largest arrays data type
- * (FCELL_TYPE or DCELL_TYPE) used by the input arrays.
- * <br><br>
- *
- * the calculations are of the following form:
- *
- * <ul>
- * <li>result = a + b -> N_ARRAY_SUM</li>
- * <li>result = a - b -> N_ARRAY_DIF</li>
- * <li>result = a * b -> N_ARRAY_MUL</li>
- * <li>result = a / b -> N_ARRAY_DIV</li>
- * </ul>
- *
- * \param a N_array_3d * - first input array
- * \param b N_array_3d * - second input array
- * \param result N_array_3d * - the optional result array
- * \param type  - the type of calculation
- * \return N_array_3d * - the pointer to the result array
- * */
-N_array_3d *N_math_array_3d(N_array_3d * a, N_array_3d * b,
-			    N_array_3d * result, int type)
-{
-    N_array_3d *c;
-    int i, j, k, setnull = 0;
-    double va = 0.0, vb = 0.0, vc = 0.0;	/*variables used for calculation */
-
-    /*Set the pointer */
-    c = result;
-
-    /*Check the array sizes */
-    if (a->cols_intern != b->cols_intern)
-	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
-    if (a->rows_intern != b->rows_intern)
-	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
-    if (a->depths_intern != b->depths_intern)
-	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
-    if (a->offset != b->offset)
-	G_fatal_error("N_math_array_3d: the arrays have different offsets");
-
-    G_debug(3, "N_math_array_3d: mathematical calculations, size: %i",
-	    a->cols_intern * a->rows_intern * a->depths_intern);
-
-    /*if the result array is null, allocate a new one, use the 
-     * largest data type of the input arrays*/
-    if (c == NULL) {
-	if (a->type == DCELL_TYPE || b->type == DCELL_TYPE) {
-	    c = N_alloc_array_3d(a->cols, a->rows, a->depths, a->offset,
-				 DCELL_TYPE);
-	    G_debug(3, "N_math_array_3d: array of type DCELL_TYPE created");
-	}
-	else {
-	    c = N_alloc_array_3d(a->cols, a->rows, a->depths, a->offset,
-				 FCELL_TYPE);
-	    G_debug(3, "N_math_array_3d: array of type FCELL_TYPE created");
-	}
-    }
-    else {
-	/*Check the array sizes */
-	if (a->cols_intern != c->cols_intern)
-	    G_fatal_error
-		("N_math_array_3d: the arrays are not of equal size");
-	if (a->rows_intern != c->rows_intern)
-	    G_fatal_error
-		("N_math_array_3d: the arrays are not of equal size");
-	if (a->depths_intern != c->depths_intern)
-	    G_fatal_error
-		("N_math_array_3d: the arrays are not of equal size");
-	if (a->offset != c->offset)
-	    G_fatal_error
-		("N_math_array_3d: the arrays have different offsets");
-    }
-
-    for (k = 0 - a->offset; k < a->depths + a->offset; k++) {
-	for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
-	    for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
-		if (!N_is_array_3d_value_null(a, i, j, k) &&
-		    !N_is_array_3d_value_null(a, i, j, k)) {
-		    /*we always calulate internally with double values */
-		    va = (double)N_get_array_3d_d_value(a, i, j, k);
-		    vb = (double)N_get_array_3d_d_value(b, i, j, k);
-		    vc = 0;
-		    setnull = 0;
-
-		    switch (type) {
-		    case N_ARRAY_SUM:
-			vc = va + vb;
-			break;
-		    case N_ARRAY_DIF:
-			vc = va - vb;
-			break;
-		    case N_ARRAY_MUL:
-			vc = va * vb;
-			break;
-		    case N_ARRAY_DIV:
-			if (vb != 0)
-			    vc = va / vb;
-			else
-			    setnull = 1;
-			break;
-		    }
-
-		    if (c->type == FCELL_TYPE) {
-			if (setnull)
-			    N_put_array_3d_value_null(c, i, j, k);
-			else
-			    N_put_array_3d_f_value(c, i, j, k, (float)vc);
-		    }
-		    if (c->type == DCELL_TYPE) {
-			if (setnull)
-			    N_put_array_3d_value_null(c, i, j, k);
-			else
-			    N_put_array_3d_d_value(c, i, j, k, vc);
-		    }
-		}
-		else {
-		    N_put_array_3d_value_null(c, i, j, k);
-		}
-	    }
-	}
-    }
-
-    return c;
-}
-
-/*!
- * \brief Convert all null values to zero values
- *
- * The complete data array inclusively offsets is used.
- *
- * \param a N_array_3d *
- * \return int - number of replaced null values
- * */
-int N_convert_array_3d_null_to_zero(N_array_3d * a)
-{
-    int i = 0, count = 0;
-
-    G_debug(3, "N_convert_array_3d_null_to_zero: convert array of size %i",
-	    a->cols_intern * a->rows_intern * a->depths_intern);
-
-    if (a->type == FCELL_TYPE)
-	for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern;
-	     i++) {
-	    if (Rast3d_is_null_value_num((void *)&(a->fcell_array[i]), FCELL_TYPE)) {
-		a->fcell_array[i] = 0.0;
-		count++;
-	    }
-	}
-
-    if (a->type == DCELL_TYPE)
-	for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern;
-	     i++) {
-	    if (Rast3d_is_null_value_num((void *)&(a->dcell_array[i]), DCELL_TYPE)) {
-		a->dcell_array[i] = 0.0;
-		count++;
-	    }
-	}
-
-
-    if (a->type == FCELL_TYPE)
-	G_debug(3,
-		"N_convert_array_3d_null_to_zero: %i values of type FCELL_TYPE are converted",
-		count);
-
-    if (a->type == DCELL_TYPE)
-	G_debug(3,
-		"N_convert_array_3d_null_to_zero: %i values of type DCELL_TYPE are converted",
-		count);
-
-    return count;
-}

Deleted: grass/trunk/lib/gpde/N_arrays_io.c
===================================================================
--- grass/trunk/lib/gpde/N_arrays_io.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_arrays_io.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,467 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:     	IO array managment functions 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-#include <math.h>
-
-#include <grass/N_pde.h>
-#include <grass/raster.h>
-#include <grass/glocale.h>
-
-
-/* ******************** 2D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Read a raster map into a N_array_2d structure
- *
- * The raster map will be opened in the current region settings.
- * If no N_array_2d structure is provided (NULL pointer), a new structure will be
- * allocated with the same data type as the raster map and the size of the current region. 
- * The array offset will be set to 0.
- * <br><br>
- * If a N_array_2d structure is provided, the values from the raster map are 
- * casted to the N_array_2d type. The array must have the same size 
- * as the current region. 
- * <br><br>
- * The new created or the provided array are returned.
- * If the reading of the raster map fails, G_fatal_error() will
- * be invoked.
- *
- * \param name * char - the name of an existing raster map
- * \param array * N_array_2d - an existing array or NULL
- * \return N_array_2d * - the existing or new allocated array
- * */
-N_array_2d *N_read_rast_to_array_2d(char *name, N_array_2d * array)
-{
-    int map;			/*The rastermap */
-    int x, y, cols, rows, type;
-    void *rast;
-    void *ptr;
-    struct Cell_head region;
-    N_array_2d *data = array;
-
-    /* Get the active region */
-    G_get_set_window(&region);
-
-    /*set the rows and cols */
-    rows = region.rows;
-    cols = region.cols;
-
-    /*open the raster map */
-    map = Rast_open_old(name, "");
-
-    type = Rast_get_map_type(map);
-
-    /*if the array is NULL create a new one with the data type of the raster map */
-    /*the offset is 0 by default */
-    if (data == NULL) {
-	if (type == DCELL_TYPE) {
-	    data = N_alloc_array_2d(cols, rows, 0, DCELL_TYPE);
-	}
-	if (type == FCELL_TYPE) {
-	    data = N_alloc_array_2d(cols, rows, 0, FCELL_TYPE);
-	}
-	if (type == CELL_TYPE) {
-	    data = N_alloc_array_2d(cols, rows, 0, CELL_TYPE);
-	}
-    }
-    else {
-	/*Check the array sizes */
-	if (data->cols != cols)
-	    G_fatal_error
-		("N_read_rast_to_array_2d: the data array size is different from the current region settings");
-	if (data->rows != rows)
-	    G_fatal_error
-		("N_read_rast_to_array_2d: the data array size is different from the current region settings");
-    }
-
-    rast = Rast_allocate_buf(type);
-
-    G_message(_("Reading raster map <%s> into memory"), name);
-
-    for (y = 0; y < rows; y++) {
-	G_percent(y, rows - 1, 10);
-
-	Rast_get_row(map, rast, y, type);
-
-	for (x = 0, ptr = rast; x < cols;
-	     x++, ptr = G_incr_void_ptr(ptr, Rast_cell_size(type))) {
-	    if (type == CELL_TYPE) {
-		if (Rast_is_c_null_value(ptr)) {
-		    N_put_array_2d_value_null(data, x, y);
-		}
-		else {
-		    if (data->type == CELL_TYPE)
-			N_put_array_2d_c_value(data, x, y,
-					       (CELL) * (CELL *) ptr);
-		    if (data->type == FCELL_TYPE)
-			N_put_array_2d_f_value(data, x, y,
-					       (FCELL) * (CELL *) ptr);
-		    if (data->type == DCELL_TYPE)
-			N_put_array_2d_d_value(data, x, y,
-					       (DCELL) * (CELL *) ptr);
-		}
-	    }
-	    if (type == FCELL_TYPE) {
-		if (Rast_is_f_null_value(ptr)) {
-		    N_put_array_2d_value_null(data, x, y);
-		}
-		else {
-		    if (data->type == CELL_TYPE)
-			N_put_array_2d_c_value(data, x, y,
-					       (CELL) * (FCELL *) ptr);
-		    if (data->type == FCELL_TYPE)
-			N_put_array_2d_f_value(data, x, y,
-					       (FCELL) * (FCELL *) ptr);
-		    if (data->type == DCELL_TYPE)
-			N_put_array_2d_d_value(data, x, y,
-					       (DCELL) * (FCELL *) ptr);
-		}
-	    }
-	    if (type == DCELL_TYPE) {
-		if (Rast_is_d_null_value(ptr)) {
-		    N_put_array_2d_value_null(data, x, y);
-		}
-		else {
-		    if (data->type == CELL_TYPE)
-			N_put_array_2d_c_value(data, x, y,
-					       (CELL) * (DCELL *) ptr);
-		    if (data->type == FCELL_TYPE)
-			N_put_array_2d_f_value(data, x, y,
-					       (FCELL) * (DCELL *) ptr);
-		    if (data->type == DCELL_TYPE)
-			N_put_array_2d_d_value(data, x, y,
-					       (DCELL) * (DCELL *) ptr);
-		}
-	    }
-	}
-    }
-
-    /* Close file */
-    Rast_close(map);
-
-    return data;
-}
-
-/*!
- * \brief Write a N_array_2d struct to a raster map
- *
- * A new raster map is created with the same type as the N_array_2d.
- * The current region is used to open the raster map.
- * The N_array_2d must have the same size as the current region.
- If the writing of the raster map fails, G_fatal_error() will
- * be invoked.
-
- * \param array N_array_2d * 
- * \param name char * - the name of the raster map
- * \return void
- *
- * */
-void N_write_array_2d_to_rast(N_array_2d * array, char *name)
-{
-    int map;			/*The rastermap */
-    int x, y, cols, rows, count, type;
-    CELL *rast = NULL;
-    FCELL *frast = NULL;
-    DCELL *drast = NULL;
-    struct Cell_head region;
-
-    if (!array)
-	G_fatal_error(_("N_array_2d * array is empty"));
-
-    /* Get the current region */
-    G_get_set_window(&region);
-
-    rows = region.rows;
-    cols = region.cols;
-    type = array->type;
-
-    /*Open the new map */
-    map = Rast_open_new(name, type);
-
-    if (type == CELL_TYPE)
-	rast = Rast_allocate_buf(type);
-    if (type == FCELL_TYPE)
-	frast = Rast_allocate_buf(type);
-    if (type == DCELL_TYPE)
-	drast = Rast_allocate_buf(type);
-
-    G_message(_("Write 2d array to raster map <%s>"), name);
-
-    count = 0;
-    for (y = 0; y < rows; y++) {
-	G_percent(y, rows - 1, 10);
-	for (x = 0; x < cols; x++) {
-	    if (type == CELL_TYPE)
-		rast[x] = N_get_array_2d_c_value(array, x, y);
-	    if (type == FCELL_TYPE)
-		frast[x] = N_get_array_2d_f_value(array, x, y);
-	    if (type == DCELL_TYPE)
-		drast[x] = N_get_array_2d_d_value(array, x, y);
-	}
-	if (type == CELL_TYPE)
-	    Rast_put_c_row(map, rast);
-	if (type == FCELL_TYPE)
-	    Rast_put_f_row(map, frast);
-	if (type == DCELL_TYPE)
-	    Rast_put_d_row(map, drast);
-    }
-
-    /* Close file */
-    Rast_close(map);
-}
-
-
-/* ******************** 3D ARRAY FUNCTIONS *********************** */
-
-/*!
- * \brief Read a volume map into a N_array_3d structure
- *
- * The volume map is opened in the current region settings.
- * If no N_array_3d structure is provided (NULL pointer), a new structure will be
- * allocated with the same data type as the volume map and the size of the current region. 
- * The array offset will be set to 0.
- * <br><br>
- *
- * If a N_array_3d structure is provided, the values from the volume map are 
- * casted to the N_array_3d type. The array must have the same size 
- * as the current region. 
- * <br><br>
- *
- * The new created or the provided array is returned.
- * If the reading of the volume map fails, Rast3d_fatal_error() will
- * be invoked.
- *
- * \param name * char - the name of an existing volume map
- * \param array * N_array_3d - an existing array or NULL
- * \param mask int - 0 = false, 1 = ture : if a mask is presenent, use it with the input volume map
- * \return N_array_3d * - the existing or new allocated array
- * */
-N_array_3d *N_read_rast3d_to_array_3d(char *name, N_array_3d * array,
-				      int mask)
-{
-    void *map = NULL;		/*The 3D Rastermap */
-    int changemask = 0;
-    int x, y, z, cols, rows, depths, type;
-    double d1 = 0, f1 = 0;
-    N_array_3d *data = array;
-    RASTER3D_Region region;
-
-
-    /*get the current region */
-    Rast3d_get_window(&region);
-
-    cols = region.cols;
-    rows = region.rows;
-    depths = region.depths;
-
-
-    if (NULL == G_find_raster3d(name, ""))
-	Rast3d_fatal_error(_("3D raster map <%s> not found"), name);
-
-    /*Open all maps with default region */
-    map =
-	Rast3d_open_cell_old(name, G_find_raster3d(name, ""), RASTER3D_DEFAULT_WINDOW,
-			RASTER3D_TILE_SAME_AS_FILE, RASTER3D_USE_CACHE_DEFAULT);
-
-    if (map == NULL)
-	Rast3d_fatal_error(_("Unable to open 3D raster map <%s>"), name);
-
-    type = Rast3d_tile_type_map(map);
-
-    /*if the array is NULL create a new one with the data type of the volume map */
-    /*the offset is 0 by default */
-    if (data == NULL) {
-	if (type == FCELL_TYPE) {
-	    data = N_alloc_array_3d(cols, rows, depths, 0, FCELL_TYPE);
-	}
-	if (type == DCELL_TYPE) {
-	    data = N_alloc_array_3d(cols, rows, depths, 0, DCELL_TYPE);
-	}
-    }
-    else {
-	/*Check the array sizes */
-	if (data->cols != cols)
-	    G_fatal_error
-		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
-	if (data->rows != rows)
-	    G_fatal_error
-		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
-	if (data->depths != depths)
-	    G_fatal_error
-		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
-    }
-
-
-    G_message(_("Read g3d map <%s> into the memory"), name);
-
-    /*if requested set the Mask on */
-    if (mask) {
-	if (Rast3d_mask_file_exists()) {
-	    changemask = 0;
-	    if (Rast3d_mask_is_off(map)) {
-		Rast3d_mask_on(map);
-		changemask = 1;
-	    }
-	}
-    }
-
-    for (z = 0; z < depths; z++) {	/*From the bottom to the top */
-	G_percent(z, depths - 1, 10);
-	for (y = 0; y < rows; y++) {
-	    for (x = 0; x < cols; x++) {
-		if (type == FCELL_TYPE) {
-		    Rast3d_get_value(map, x, y, z, &f1, type);
-		    if (Rast_is_f_null_value((void *)&f1)) {
-			N_put_array_3d_value_null(data, x, y, z);
-		    }
-		    else {
-			if (data->type == FCELL_TYPE)
-			    N_put_array_3d_f_value(data, x, y, z, f1);
-			if (data->type == DCELL_TYPE)
-			    N_put_array_3d_d_value(data, x, y, z, (double)f1);
-		    }
-		}
-		else {
-		    Rast3d_get_value(map, x, y, z, &d1, type);
-		    if (Rast_is_d_null_value((void *)&d1)) {
-			N_put_array_3d_value_null(data, x, y, z);
-		    }
-		    else {
-			if (data->type == FCELL_TYPE)
-			    N_put_array_3d_f_value(data, x, y, z, (float)d1);
-			if (data->type == DCELL_TYPE)
-			    N_put_array_3d_d_value(data, x, y, z, d1);
-		    }
-
-		}
-	    }
-	}
-    }
-
-    /*We set the Mask off, if it was off before */
-    if (mask) {
-	if (Rast3d_mask_file_exists())
-	    if (Rast3d_mask_is_on(map) && changemask)
-		Rast3d_mask_off(map);
-    }
-
-    /* Close files and exit */
-    if (!Rast3d_close(map))
-	Rast3d_fatal_error(map, NULL, 0, _("Error closing g3d file"));
-
-    return data;
-}
-
-/*!
- * \brief Write a N_array_3d struct to a volume map
- *
- * A new volume map is created with the same type as the N_array_3d.
- * The current region is used to open the volume map.
- * The N_array_3d must have the same size as the current region.
- * If the writing of the volume map fails, Rast3d_fatal_error() will
- * be invoked.
- *
- *
- * \param array N_array_3d * 
- * \param name char * - the name of the volume map
- * \param mask int - 1 = use a 3d mask, 0 do not use a 3d mask
- * \return void
- *
- * */
-void N_write_array_3d_to_rast3d(N_array_3d * array, char *name, int mask)
-{
-    void *map = NULL;		/*The 3D Rastermap */
-    int changemask = 0;
-    int x, y, z, cols, rows, depths, count, type;
-    double d1 = 0.0, f1 = 0.0;
-    N_array_3d *data = array;
-    RASTER3D_Region region;
-
-    /*get the current region */
-    Rast3d_get_window(&region);
-
-    cols = region.cols;
-    rows = region.rows;
-    depths = region.depths;
-    type = data->type;
-
-    /*Check the array sizes */
-    if (data->cols != cols)
-	G_fatal_error
-	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
-    if (data->rows != rows)
-	G_fatal_error
-	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
-    if (data->depths != depths)
-	G_fatal_error
-	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
-
-    /*Open the new map */
-    if (type == DCELL_TYPE)
-        map = Rast3d_open_new_opt_tile_size(name, RASTER3D_USE_CACHE_XY, &region, DCELL_TYPE, 32);
-    else if (type == FCELL_TYPE)
-        map = Rast3d_open_new_opt_tile_size(name, RASTER3D_USE_CACHE_XY, &region, FCELL_TYPE, 32);
-
-    if (map == NULL)
-	Rast3d_fatal_error(_("Error opening g3d map <%s>"), name);
-
-    G_message(_("Write 3d array to g3d map <%s>"), name);
-
-    /*if requested set the Mask on */
-    if (mask) {
-	if (Rast3d_mask_file_exists()) {
-	    changemask = 0;
-	    if (Rast3d_mask_is_off(map)) {
-		Rast3d_mask_on(map);
-		changemask = 1;
-	    }
-	}
-    }
-
-    count = 0;
-    for (z = 0; z < depths; z++) {	/*From the bottom to the top */
-	G_percent(z, depths - 1, 10);
-	for (y = 0; y < rows; y++) {
-	    for (x = 0; x < cols; x++) {
-		if (type == FCELL_TYPE) {
-		    f1 = N_get_array_3d_f_value(data, x, y, z);
-		    Rast3d_put_float(map, x, y, z, f1);
-		}
-		else if (type == DCELL_TYPE) {
-		    d1 = N_get_array_3d_d_value(data, x, y, z);
-		    Rast3d_put_double(map, x, y, z, d1);
-		}
-	    }
-	}
-    }
-
-    /*We set the Mask off, if it was off before */
-    if (mask) {
-	if (Rast3d_mask_file_exists())
-	    if (Rast3d_mask_is_on(map) && changemask)
-		Rast3d_mask_off(map);
-    }
-
-    /* Flush all tile */
-    if (!Rast3d_flush_all_tiles(map))
-	Rast3d_fatal_error("Error flushing tiles with Rast3d_flush_all_tiles");
-    /* Close files and exit */
-    if (!Rast3d_close(map))
-	Rast3d_fatal_error(map, NULL, 0, _("Error closing g3d file"));
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_geom.c
===================================================================
--- grass/trunk/lib/gpde/N_geom.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_geom.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,204 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      part of the gpde library
-* 		allocation, destroing and initializing the geometric struct
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-
-#include <grass/N_pde.h>
-
-/* *************************************************************** * 
- * *********** Konstruktor *************************************** * 
- * *************************************************************** */
-/*!
- * \brief Allocate the pde geometry data structure and return a pointer to the new allocated structure
- *
- * \return N_geom_data *
- * */
-N_geom_data *N_alloc_geom_data(void)
-{
-    N_geom_data *geom = (N_geom_data *) G_calloc(1, sizeof(N_geom_data));
-
-    geom->area = NULL;
-    geom->planimetric = 1;
-    geom->dim = 0;
-
-    return geom;
-}
-
-/* *************************************************************** * 
- * *********** Destruktor **************************************** * 
- * *************************************************************** */
-/*!
- * \brief Release memory of a pde geometry data structure
- *
- * \param  geom N_geom_data *
- * \return void
- * */
-void N_free_geom_data(N_geom_data * geom)
-{
-    if (geom->area != NULL)
-	G_free(geom->area);
-
-    G_free(geom);
-    return;
-}
-
-/* *************************************************************** * 
- * *************************************************************** * 
- * *************************************************************** */
-/*!
- * \brief Initiate a pde geometry data structure with a 3d region
- *
- * If the projection is not planimetric, a double array will be created based on the 
- * number of rows of the provided region
- *
- * \param region3d RASTER3D_Region *
- * \param geodata N_geom_data * - if a NULL pointer is given, a new structure will be allocatet and returned
- *
- * \return N_geom_data *
- * */
-N_geom_data *N_init_geom_data_3d(RASTER3D_Region * region3d, N_geom_data * geodata)
-{
-    N_geom_data *geom = geodata;
-    struct Cell_head region2d;
-
-#pragma omp critical
-    {
-
-	G_debug(2,
-		"N_init_geom_data_3d: initializing the geometry structure");
-
-	if (geom == NULL)
-	    geom = N_alloc_geom_data();
-
-	geom->dz = region3d->tb_res * G_database_units_to_meters_factor();	/*this function is not thread safe */
-	geom->depths = region3d->depths;
-	geom->dim = 3;
-
-	/*convert the 3d into a 2d region and begin the area calculation */
-	G_get_set_window(&region2d);	/*this function is not thread safe */
-	Rast3d_region_to_cell_head(region3d, &region2d);
-    }
-
-    return N_init_geom_data_2d(&region2d, geom);
-}
-
-
-/* *************************************************************** * 
- * *************************************************************** * 
- * *************************************************************** */
-/*!
- * \brief Initiate a pde geometry data structure with a 2d region
- *
- * If the projection is not planimetric, a double array will be created based on the 
- * number of rows of the provided region storing all computed areas for each row
- *
- * \param region sruct Cell_head *
- * \param geodata N_geom_data * - if a NULL pointer is given, a new structure will be allocatet and returned
- *
- * \return N_geom_data *
- * */
-N_geom_data *N_init_geom_data_2d(struct Cell_head * region,
-				 N_geom_data * geodata)
-{
-    N_geom_data *geom = geodata;
-    struct Cell_head backup;
-    double meters;
-    short ll = 0;
-    int i;
-
-
-    /*create an openmp lock to assure that only one thread at a time will access this function */
-#pragma omp critical
-    {
-	G_debug(2,
-		"N_init_geom_data_2d: initializing the geometry structure");
-
-	/*make a backup from this region */
-	G_get_set_window(&backup);	/*this function is not thread safe */
-	/*set the current region */
-	Rast_set_window(region);	/*this function is not thread safe */
-
-	if (geom == NULL)
-	    geom = N_alloc_geom_data();
-
-	meters = G_database_units_to_meters_factor();	/*this function is not thread safe */
-
-	/*set the dim to 2d if it was not initiated with 3, thats a bit ugly :( */
-	if (geom->dim != 3)
-	    geom->dim = 2;
-
-	geom->planimetric = 1;
-	geom->rows = region->rows;
-	geom->cols = region->cols;
-	geom->dx = region->ew_res * meters;
-	geom->dy = region->ns_res * meters;
-	geom->Az = geom->dy * geom->dx;	/*square meters in planimetric proj */
-	/*depths and dz are initialized with a 3d region */
-
-	/*Begin the area calculation */
-	ll = G_begin_cell_area_calculations();	/*this function is not thread safe */
-
-	/*if the projection is not planimetric, calc the area for each row */
-	if (ll == 2) {
-	    G_debug(2,
-		    "N_init_geom_data_2d: calculating the areas for non parametric projection");
-	    geom->planimetric = 0;
-
-	    if (geom->area != NULL)
-		G_free(geom->area);
-	    else
-		geom->area = G_calloc(geom->rows, sizeof(double));
-
-	    /*fill the area vector */
-	    for (i = 0; i < geom->rows; i++) {
-		geom->area[i] = G_area_of_cell_at_row(i);	/*square meters */
-	    }
-	}
-
-	/*restore the old region */
-	Rast_set_window(&backup);	/*this function is not thread safe */
-    }
-
-    return geom;
-}
-
-/* *************************************************************** * 
- * *************************************************************** * 
- * *************************************************************** */
-/*!
- * \brief Get the areay size in square meter of one cell (x*y) at row
- *
- * This function works for two and three dimensions
- *
- * \param geom N_geom_data *
- * \param row int
- * \return area double
- *
- * */
-double N_get_geom_data_area_of_cell(N_geom_data * geom, int row)
-{
-    if (geom->planimetric) {
-	G_debug(6, "N_get_geom_data_area_of_cell: %g", geom->Az);
-	return geom->Az;
-    }
-    else {
-	G_debug(6, "N_get_geom_data_area_of_cell: %g", geom->area[row]);
-	return geom->area[row];
-    }
-
-    return 0.0;
-}

Deleted: grass/trunk/lib/gpde/N_gradient.c
===================================================================
--- grass/trunk/lib/gpde/N_gradient.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_gradient.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,1113 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:     	gradient management functions 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <grass/N_pde.h>
-
-/*!
- * \brief Allocate a N_gradient_2d structure
- *
- * \return N_gradient_2d *
- *
- * */
-N_gradient_2d *N_alloc_gradient_2d(void)
-{
-    N_gradient_2d *grad;
-
-    grad = (N_gradient_2d *) G_calloc(1, sizeof(N_gradient_2d));
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_2d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_2d(N_gradient_2d * grad)
-{
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-/*!
- * \brief allocate and initialize a N_gradient_2d structure
- *
- * \param NC double - the gradient between northern and center cell
- * \param SC double - the gradient between southern and center cell
- * \param WC double - the gradient between western and center cell
- * \param EC double - the gradient between eastern and center cell
- * \return N_gradient_2d *
- *
- * */
-N_gradient_2d *N_create_gradient_2d(double NC, double SC, double WC,
-				    double EC)
-{
-    N_gradient_2d *grad;
-
-    G_debug(5, "N_create_gradient_2d: create N_gradient_2d");
-
-    grad = N_alloc_gradient_2d();
-
-    grad->NC = NC;
-    grad->SC = SC;
-    grad->WC = WC;
-    grad->EC = EC;
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_2d structure
- *
- * \param source - the source N_gradient_2d struct
- * \param target - the target N_gradient_2d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int N_copy_gradient_2d(N_gradient_2d * source, N_gradient_2d * target)
-{
-    G_debug(5, "N_copy_gradient_2d: copy N_gradient_2d");
-
-    if (!source || !target)
-	return 0;
-
-    target->NC = source->NC;
-    target->SC = source->SC;
-    target->WC = source->WC;
-    target->EC = source->EC;
-
-    return 1;
-}
-
-/*!
- * \brief Return a N_gradient_2d structure calculated from the input gradient field
- * at position [row][col]
- *
- *  This function returns the gradient of a cell at position [row][col] from the input gradient field.
- *  Returend is a new structure of type N_gradient_2d.
- *
- *  \param field N_gradient_field_2d * - A two dimensional gradient field
- *  \param gradient N_gradient_2d * - the gradient structure which should be filled with data, if a NULL pointer is given, a new structure will be created
- *  \param col int
- *  \param row int
- *  \return N_gradient_2d * - the new or filled gradient structure
- *  
- *
- * */
-N_gradient_2d *N_get_gradient_2d(N_gradient_field_2d * field,
-				 N_gradient_2d * gradient, int col, int row)
-{
-    double NC = 0, SC = 0, WC = 0, EC = 0;
-    N_gradient_2d *grad = gradient;
-
-
-    NC = N_get_array_2d_d_value(field->y_array, col, row);
-    SC = N_get_array_2d_d_value(field->y_array, col, row + 1);
-    WC = N_get_array_2d_d_value(field->x_array, col, row);
-    EC = N_get_array_2d_d_value(field->x_array, col + 1, row);
-
-    G_debug(5,
-	    "N_get_gradient_2d: calculate N_gradient_2d NC %g SC %g WC %g EC %g",
-	    NC, SC, WC, EC);
-
-    /*if gradient is a NULL pointer, create a new one */
-    if (!grad) {
-	grad = N_create_gradient_2d(NC, SC, WC, EC);
-    }
-    else {
-	grad->NC = NC;
-	grad->SC = SC;
-	grad->WC = WC;
-	grad->EC = EC;
-    }
-
-    return grad;
-}
-
-/*!
- * \brief Allocate a N_gradient_3d structure
- *
- * \return N_gradient_3d *
- *
- * */
-N_gradient_3d *N_alloc_gradient_3d(void)
-{
-    N_gradient_3d *grad;
-
-    grad = (N_gradient_3d *) G_calloc(1, sizeof(N_gradient_3d));
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_3d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_3d(N_gradient_3d * grad)
-{
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-
-/*!
- * \brief allocate and initialize a N_gradient_3d structure
- *
- * \param NC double - the gradient between northern and center cell
- * \param SC double - the gradient between southern and center cell
- * \param WC double - the gradient between western and center cell
- * \param EC double - the gradient between eastern and center cell
- * \param TC double - the gradient between top and center cell
- * \param BC double - the gradient between bottom and center cell
- * \return N_gradient_3d *
- *
- * */
-N_gradient_3d *N_create_gradient_3d(double NC, double SC, double WC,
-				    double EC, double TC, double BC)
-{
-    N_gradient_3d *grad;
-
-    G_debug(5, "N_create_gradient_3d: create N_gradient_3d");
-
-    grad = N_alloc_gradient_3d();
-
-    grad->NC = NC;
-    grad->SC = SC;
-    grad->WC = WC;
-    grad->EC = EC;
-    grad->TC = TC;
-    grad->BC = BC;
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_3d structure
- *
- * \param source - the source N_gradient_3d struct
- * \param target - the target N_gradient_3d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int N_copy_gradient_3d(N_gradient_3d * source, N_gradient_3d * target)
-{
-    G_debug(5, "N_copy_gradient_3d: copy N_gradient_3d");
-
-    if (!source || !target)
-	return 0;
-
-    target->NC = source->NC;
-    target->SC = source->SC;
-    target->WC = source->WC;
-    target->EC = source->EC;
-    target->TC = source->TC;
-    target->BC = source->BC;
-
-    return 1;
-}
-
-
-/*!
- * \brief Return a N_gradient_3d structure calculated from the input gradient field
- * at position [depth][row][col]
- *
- *  This function returns the gradient of a 3d cell at position [depth][row][col] from the input gradient field.
- *  Returned is a new structure of type N_gradient_3d.
- *
- *  \param field N_gradient_field_3d * - A three dimensional gradient field
- *  \param gradient N_gradient_3d * - an existing gradient structure or a NULL pointer, if a NULL pointer is providet a new structure will be returned
- *  \param col int
- *  \param row int
- *  \param depth int
- *  \return N_gradient_3d *
- *  
- *
- * */
-N_gradient_3d *N_get_gradient_3d(N_gradient_field_3d * field,
-				 N_gradient_3d * gradient, int col, int row,
-				 int depth)
-{
-    double NC, SC, WC, EC, TC, BC;
-    N_gradient_3d *grad = gradient;
-
-    NC = N_get_array_3d_d_value(field->y_array, col, row, depth);
-    SC = N_get_array_3d_d_value(field->y_array, col, row + 1, depth);
-    WC = N_get_array_3d_d_value(field->x_array, col, row, depth);
-    EC = N_get_array_3d_d_value(field->x_array, col + 1, row, depth);
-    BC = N_get_array_3d_d_value(field->z_array, col, row, depth);
-    TC = N_get_array_3d_d_value(field->z_array, col, row, depth + 1);
-
-    G_debug(6,
-	    "N_get_gradient_3d: calculate N_gradient_3d NC %g SC %g WC %g EC %g TC %g BC %g",
-	    NC, SC, WC, EC, TC, BC);
-
-    /*if gradient is a NULL pointer, create a new one */
-    if (!grad) {
-	grad = N_create_gradient_3d(NC, SC, WC, EC, TC, BC);
-    }
-    else {
-	grad->NC = NC;
-	grad->SC = SC;
-	grad->WC = WC;
-	grad->EC = EC;
-	grad->BC = BC;
-	grad->TC = TC;
-    }
-
-    return grad;
-}
-
-/*!
- * \brief Allocate a N_gradient_neighbours_x structure
- *
- * This structure contains all neighbour gradients in x direction of one cell  
- *
- * \return N_gradient_neighbours_x  *
- *
- * */
-N_gradient_neighbours_x *N_alloc_gradient_neighbours_x(void)
-{
-    N_gradient_neighbours_x *grad;
-
-    grad =
-	(N_gradient_neighbours_x *) G_calloc(1,
-					     sizeof(N_gradient_neighbours_x));
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_x structure
- *
- * \return void
- *
- * */
-void N_free_gradient_neighbours_x(N_gradient_neighbours_x * grad)
-{
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-
-/*!
- * \brief Allocate and initialize a N_gradient_neighbours_x structure
- *
- * \param NWN double - the gradient between north-west and northern cell
- * \param NEN double - the gradient between north-east and northern cell
- * \param WC double - the gradient between western and center cell
- * \param EC double - the gradient between eastern and center cell
- * \param SWS double - the gradient between south-west and southern cell
- * \param SES double - the gradient between south-east and southern cell
- * \return N_gradient_neighbours_x *
-
- *
- * */
-N_gradient_neighbours_x *N_create_gradient_neighbours_x(double NWN,
-							double NEN, double WC,
-							double EC, double SWS,
-							double SES)
-{
-    N_gradient_neighbours_x *grad;
-
-    G_debug(6,
-	    "N_create_gradient_neighbours_x: create N_gradient_neighbours_x");
-
-    grad = N_alloc_gradient_neighbours_x();
-
-    grad->NWN = NWN;
-    grad->NEN = NEN;
-    grad->WC = WC;
-    grad->EC = EC;
-    grad->SWS = SWS;
-    grad->SES = SES;
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_neighbours_x structure
- *
- * \param source - the source N_gradient_neighbours_x struct
- * \param target - the target N_gradient_neighbours_x struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_neighbours_x(N_gradient_neighbours_x * source,
-			     N_gradient_neighbours_x * target)
-{
-    G_debug(6, "N_copy_gradient_neighbours_x: copy N_gradient_neighbours_x");
-
-    if (!source || !target)
-	return 0;
-
-    target->NWN = source->NWN;
-    target->NEN = source->NEN;
-    target->WC = source->WC;
-    target->EC = source->EC;
-    target->SWS = source->SWS;
-    target->SES = source->SES;
-
-    return 1;
-}
-
-/*!
- * \brief Allocate a N_gradient_neighbours_y structure
- *
- * This structure contains all neighbour gradients in y direction of one cell  
- *
- * \return N_gradient_neighbours_y  *
- *
- * */
-N_gradient_neighbours_y *N_alloc_gradient_neighbours_y(void)
-{
-    N_gradient_neighbours_y *grad;
-
-    grad =
-	(N_gradient_neighbours_y *) G_calloc(1,
-					     sizeof(N_gradient_neighbours_y));
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_y structure
- *
- * \return void
- *
- * */
-void N_free_gradient_neighbours_y(N_gradient_neighbours_y * grad)
-{
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-/*!
- * \brief Allocate and initialize a N_gradient_neighbours_y structure
- *
- * \param NWW double - the gradient between north-west and western cell
- * \param NEE double - the gradient between north-east and eastern cell
- * \param NC double - the gradient between northern and center cell
- * \param SC double - the gradient between southern and center cell
- * \param SWW double - the gradient between south-west and western cell
- * \param SEE double - the gradient between south-east and eastern cell
- * \return N_gradient_neighbours_y *
-
- *
- * */
-N_gradient_neighbours_y *N_create_gradient_neighbours_y(double NWW,
-							double NEE, double NC,
-							double SC, double SWW,
-							double SEE)
-{
-    N_gradient_neighbours_y *grad;
-
-    G_debug(6,
-	    "N_create_gradient_neighbours_y: create N_gradient_neighbours_y");
-
-    grad = N_alloc_gradient_neighbours_y();
-
-    grad->NWW = NWW;
-    grad->NEE = NEE;
-    grad->NC = NC;
-    grad->SC = SC;
-    grad->SWW = SWW;
-    grad->SEE = SEE;
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_neighbours_y structure
- *
- * \param source - the source N_gradient_neighbours_y struct
- * \param target - the target N_gradient_neighbours_y struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_neighbours_y(N_gradient_neighbours_y * source,
-			     N_gradient_neighbours_y * target)
-{
-    G_debug(6, "N_copy_gradient_neighbours_y: copy N_gradient_neighbours_y");
-
-    if (!source || !target)
-	return 0;
-
-    target->NWW = source->NWW;
-    target->NEE = source->NEE;
-    target->NC = source->NC;
-    target->SC = source->SC;
-    target->SWW = source->SWW;
-    target->SEE = source->SEE;
-
-    return 1;
-}
-
-/*!
- * \brief Allocate a N_gradient_neighbours_z structure
- *
- * This structure contains all neighbour gradients in z direction of one cell  
- *
- * \return N_gradient_neighbours_z  *
- *
- * */
-N_gradient_neighbours_z *N_alloc_gradient_neighbours_z(void)
-{
-    N_gradient_neighbours_z *grad;
-
-    grad =
-	(N_gradient_neighbours_z *) G_calloc(1,
-					     sizeof(N_gradient_neighbours_z));
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_z structure
- *
- * \return void
- *
- * */
-void N_free_gradient_neighbours_z(N_gradient_neighbours_z * grad)
-{
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-/*!
- * \brief Allocate and initialize a N_gradient_neighbours_z structure
- *
- * \param NWZ double - the gradient between upper and lower north-western cells
- * \param NZ double - the gradient between upper and lower northern cells
- * \param NEZ double - the gradient between upper and lower north-eastern cells
- * \param WZ double - the gradient between upper and lower western cells
- * \param CZ double - the gradient between upper and lower center cells
- * \param EZ double - the gradient between upper and lower eastern cells
- * \param SWZ double - the gradient between upper and lower south-western cells
- * \param SZ double - the gradient between upper and lower southern cells
- * \param SEZ double - the gradient between upper and lower south-eastern cells
- * \return N_gradient_neighbours_z *
-
- *
- * */
-N_gradient_neighbours_z *N_create_gradient_neighbours_z(double NWZ, double NZ,
-							double NEZ, double WZ,
-							double CZ, double EZ,
-							double SWZ, double SZ,
-							double SEZ)
-{
-    N_gradient_neighbours_z *grad;
-
-    G_debug(6,
-	    "N_create_gradient_neighbours_z: create N_gradient_neighbours_z");
-
-    grad = N_alloc_gradient_neighbours_z();
-
-    grad->NWZ = NWZ;
-    grad->NZ = NZ;
-    grad->NEZ = NEZ;
-    grad->WZ = WZ;
-    grad->CZ = CZ;
-    grad->EZ = EZ;
-    grad->SWZ = SWZ;
-    grad->SZ = SZ;
-    grad->SEZ = SEZ;
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_neighbours_z structure
- *
- * \param source - the source N_gradient_neighbours_z struct
- * \param target - the target N_gradient_neighbours_z struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_neighbours_z(N_gradient_neighbours_z * source,
-			     N_gradient_neighbours_z * target)
-{
-    G_debug(6, "N_copy_gradient_neighbours_z: copy N_gradient_neighbours_z");
-
-    if (!source || !target)
-	return 0;
-
-    target->NWZ = source->NWZ;
-    target->NZ = source->NZ;
-    target->NEZ = source->NEZ;
-    target->WZ = source->WZ;
-    target->CZ = source->CZ;
-    target->EZ = source->EZ;
-    target->SWZ = source->SWZ;
-    target->SZ = source->SZ;
-    target->SEZ = source->SEZ;
-
-    return 1;
-}
-
-/*!
- * \brief Allocate a N_gradient_neighbours_2d structure
- *
- * This structure contains all neighbour gradients in all directions of one cell 
- * in a 2d raster layer
- *
- * \return N_gradient_neighbours_2d *
- *
- * */
-N_gradient_neighbours_2d *N_alloc_gradient_neighbours_2d(void)
-{
-    N_gradient_neighbours_2d *grad;
-
-    grad =
-	(N_gradient_neighbours_2d *) G_calloc(1,
-					      sizeof
-					      (N_gradient_neighbours_2d));
-
-    grad->x = N_alloc_gradient_neighbours_x();
-    grad->y = N_alloc_gradient_neighbours_y();
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_2d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_neighbours_2d(N_gradient_neighbours_2d * grad)
-{
-
-    N_free_gradient_neighbours_x(grad->x);
-    N_free_gradient_neighbours_y(grad->y);
-
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-/*!
- * \brief Allocate and initialize a N_gradient_neighbours_2d structure
- *
- * The parameter N_gradient_neighbours x and y are copied into the new allocated structure 
- * and can be deleted after the initializing
- *
- * \return N_gradient_neighbours_2d * -- if failure NULL is returned
- *
- * */
-N_gradient_neighbours_2d
-    * N_create_gradient_neighbours_2d(N_gradient_neighbours_x * x,
-				      N_gradient_neighbours_y * y)
-{
-    N_gradient_neighbours_2d *grad;
-    int fail = 0;
-
-    G_debug(5,
-	    "N_create_gradient_neighbours_2d: create N_gradient_neighbours_2d");
-
-    grad = N_alloc_gradient_neighbours_2d();
-
-    if (!N_copy_gradient_neighbours_x(x, grad->x))
-	fail++;
-    if (!N_copy_gradient_neighbours_y(y, grad->y))
-	fail++;
-
-    if (fail > 0) {
-	N_free_gradient_neighbours_2d(grad);
-	grad = NULL;
-    }
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_neighbours_2d structure
- *
- * \param source - the source N_gradient_neighbours_2d struct
- * \param target - the target N_gradient_neighbours_2d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_neighbours_2d(N_gradient_neighbours_2d * source,
-			      N_gradient_neighbours_2d * target)
-{
-    int fail = 0;
-
-    G_debug(5,
-	    "N_copy_gradient_neighbours_2d: copy N_gradient_neighbours_2d");
-
-    if (!source || !target)
-	return 0;
-
-    if (!(N_copy_gradient_neighbours_x(source->x, target->x)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(source->y, target->y)))
-	fail++;
-
-    if (fail > 0) {
-	return 0;
-    }
-
-    return 1;
-}
-
-/*!
- * \brief Return a N_gradient_neighbours_2d structure calculated from the input gradient field
- * at position [row][col]
- *
- *  This function returns the gradient neighbours in x and y dierection 
- *  of a cell at position [row][col] from the input gradient field.
- *  Returend is a pointer to a structure of type N_gradient_neighbours_2d.
- *
- *  \param field N_gradient_field_2d * - A two dimensional gradient field
- *  \param gradient N_gradient_neighbours_2d * - the gradient structure which should be filled with data, if a NULL pointer is given, a new structure will be created
- *  \param col int
- *  \param row int
- *  \return N_gradient_neighbours_2d * - the new or filled gradient structure
- *  
- *
- * */
-N_gradient_neighbours_2d *N_get_gradient_neighbours_2d(N_gradient_field_2d *
-						       field,
-						       N_gradient_neighbours_2d
-						       * gradient, int col,
-						       int row)
-{
-    double NWN, NEN, WC, EC, SWS, SES;
-    double NWW, NEE, NC, SC, SWW, SEE;
-    N_gradient_neighbours_2d *grad = NULL;
-    N_gradient_neighbours_x *grad_x = NULL;
-    N_gradient_neighbours_y *grad_y = NULL;
-
-
-    NWN = N_get_array_2d_d_value(field->x_array, col, row - 1);
-    NEN = N_get_array_2d_d_value(field->x_array, col + 1, row - 1);
-    WC = N_get_array_2d_d_value(field->x_array, col, row);
-    EC = N_get_array_2d_d_value(field->x_array, col + 1, row);
-    SWS = N_get_array_2d_d_value(field->x_array, col, row + 1);
-    SES = N_get_array_2d_d_value(field->x_array, col + 1, row + 1);
-
-    NWW = N_get_array_2d_d_value(field->y_array, col - 1, row);
-    NEE = N_get_array_2d_d_value(field->y_array, col + 1, row);
-    NC = N_get_array_2d_d_value(field->y_array, col, row);
-    SC = N_get_array_2d_d_value(field->y_array, col, row + 1);
-    SWW = N_get_array_2d_d_value(field->y_array, col - 1, row + 1);
-    SEE = N_get_array_2d_d_value(field->y_array, col + 1, row + 1);
-
-
-    grad_x = N_create_gradient_neighbours_x(NWN, NEN, WC, EC, SWS, SES);
-    grad_y = N_create_gradient_neighbours_y(NWW, NEE, NC, SC, SWW, SEE);
-
-    G_debug(5,
-	    "N_get_gradient_neighbours_2d: calculate N_gradient_neighbours_x NWN %g NEN %g WC %g EC %g SWS %g SES %g",
-	    NWN, NEN, WC, EC, SWS, SES);
-
-    G_debug(5,
-	    "N_get_gradient_neighbours_2d: calculate N_gradient_neighbours_y NWW %g NEE %g NC %g SC %g SWW %g SEE %g",
-	    NWW, NEE, NC, SC, SWW, SEE);
-
-
-    /*if gradient is a NULL pointer, create a new one */
-    if (!gradient) {
-	grad = N_create_gradient_neighbours_2d(grad_x, grad_y);
-	gradient = grad;
-    }
-    else {
-	grad = N_create_gradient_neighbours_2d(grad_x, grad_y);
-	N_copy_gradient_neighbours_2d(grad, gradient);
-	N_free_gradient_neighbours_2d(grad);
-    }
-
-    N_free_gradient_neighbours_x(grad_x);
-    N_free_gradient_neighbours_y(grad_y);
-
-    return gradient;
-}
-
-
-/*!
- * \brief Allocate a N_gradient_neighbours_3d structure
- *
- * This structure contains all neighbour gradients in all directions of one cell 
- * in a 3d raster layer
- *
- * \return N_gradient_neighbours_3d *
- *
- * */
-N_gradient_neighbours_3d *N_alloc_gradient_neighbours_3d(void)
-{
-    N_gradient_neighbours_3d *grad;
-
-    grad =
-	(N_gradient_neighbours_3d *) G_calloc(1,
-					      sizeof
-					      (N_gradient_neighbours_3d));
-
-    grad->xt = N_alloc_gradient_neighbours_x();
-    grad->xc = N_alloc_gradient_neighbours_x();
-    grad->xb = N_alloc_gradient_neighbours_x();
-    grad->yt = N_alloc_gradient_neighbours_y();
-    grad->yc = N_alloc_gradient_neighbours_y();
-    grad->yb = N_alloc_gradient_neighbours_y();
-    grad->zt = N_alloc_gradient_neighbours_z();
-    grad->zb = N_alloc_gradient_neighbours_z();
-
-    return grad;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_3d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_neighbours_3d(N_gradient_neighbours_3d * grad)
-{
-
-    N_free_gradient_neighbours_x(grad->xt);
-    N_free_gradient_neighbours_x(grad->xc);
-    N_free_gradient_neighbours_x(grad->xb);
-    N_free_gradient_neighbours_y(grad->yt);
-    N_free_gradient_neighbours_y(grad->yc);
-    N_free_gradient_neighbours_y(grad->yb);
-    N_free_gradient_neighbours_z(grad->zt);
-    N_free_gradient_neighbours_z(grad->zb);
-
-    G_free(grad);
-    grad = NULL;
-
-    return;
-}
-
-/*!
- * \brief Allocate and initialize a N_gradient_neighbours_3d structure
- *
- * The parameter N_gradient_neighbours x(tcb) and y(tcb) and z(tb) are copied into the new allocated structure 
- * and can be deleted after the initializing
- *
- * \return N_gradient_neighbours_3d * -- if failure NULL is returned
-
- *
- * */
-N_gradient_neighbours_3d
-    * N_create_gradient_neighbours_3d(N_gradient_neighbours_x * xt,
-				      N_gradient_neighbours_x * xc,
-				      N_gradient_neighbours_x * xb,
-				      N_gradient_neighbours_y * yt,
-				      N_gradient_neighbours_y * yc,
-				      N_gradient_neighbours_y * yb,
-				      N_gradient_neighbours_z * zt,
-				      N_gradient_neighbours_z * zb)
-{
-    N_gradient_neighbours_3d *grad;
-    int fail = 0;
-
-    G_debug(5,
-	    "N_create_gradient_neighbours_3d: create N_gradient_neighbours_3d");
-
-    grad = N_alloc_gradient_neighbours_3d();
-
-    if (!(N_copy_gradient_neighbours_x(xt, grad->xt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_x(xc, grad->xc)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_x(xb, grad->xb)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(yt, grad->yt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(yc, grad->yc)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(yb, grad->yb)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_z(zt, grad->zt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_z(zb, grad->zb)))
-	fail++;
-
-    if (fail > 0) {
-	return NULL;
-    }
-
-    return grad;
-}
-
-/*!
- * \brief copy a N_gradient_neighbours_3d structure
- *
- * \param source - the source N_gradient_neighbours_3d struct
- * \param target - the target N_gradient_neighbours_3d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_neighbours_3d(N_gradient_neighbours_3d * source,
-			      N_gradient_neighbours_3d * target)
-{
-    int fail = 0;
-
-    G_debug(5,
-	    "N_copy_gradient_neighbours_3d: copy N_gradient_neighbours_3d");
-
-    if (!source || !target)
-	return 0;
-
-    if (!(N_copy_gradient_neighbours_x(source->xt, target->xt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_x(source->xc, target->xc)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_x(source->xb, target->xb)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(source->yt, target->yt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(source->yc, target->yc)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_y(source->yb, target->yb)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_z(source->zt, target->zt)))
-	fail++;
-    if (!(N_copy_gradient_neighbours_z(source->zb, target->zb)))
-	fail++;
-
-    if (fail > 0) {
-	return 0;
-    }
-
-    return 1;
-}
-
-/*!
- * \brief Allocate a N_gradient_field_2d
- *
- * The field arrays are of type DCELL. 
- *
- * \param rows - number of rows of the 2d array from which the gradient should be calculated
- * \param cols - number of cols of the 2d array from which the gradient should be calculated
- * \return N_gradient_field_2d *
- *
- * */
-N_gradient_field_2d *N_alloc_gradient_field_2d(int cols, int rows)
-{
-    N_gradient_field_2d *field;
-
-    G_debug(5,
-	    "N_alloc_gradient_field_2d: allocate a N_gradient_field_2d struct");
-
-    field = (N_gradient_field_2d *) G_calloc(1, sizeof(N_gradient_field_2d));
-
-    field->x_array = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    field->y_array = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-
-    field->cols = cols;
-    field->rows = rows;
-
-    return field;
-}
-
-/*!
- * \brief Free's a N_gradient_neighbours_2d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_field_2d(N_gradient_field_2d * field)
-{
-
-    N_free_array_2d(field->x_array);
-    N_free_array_2d(field->y_array);
-
-    G_free(field);
-
-    field = NULL;
-
-    return;
-}
-
-/*!
- * \brief Copy N_gradient_field_2d structure from source to target
- *
- * \param source - the source N_gradient_field_2d struct
- * \param target - the target N_gradient_field_2d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_field_2d(N_gradient_field_2d * source,
-			 N_gradient_field_2d * target)
-{
-    G_debug(3, "N_copy_gradient_field_2d: copy N_gradient_field_2d");
-
-    if (!source || !target)
-	return 0;
-
-    N_copy_array_2d(source->x_array, target->x_array);
-    N_copy_array_2d(source->y_array, target->y_array);
-
-    return 1;
-}
-
-/*! \brief Print gradient field information to stdout
- *
- * \param field N_gradient_2d_field *
- * \return void
- *
- * */
-void N_print_gradient_field_2d_info(N_gradient_field_2d * field)
-{
-    fprintf(stdout, "N_gradient_field_2d \n");
-    fprintf(stdout, "Cols %i\n", field->cols);
-    fprintf(stdout, "Rows: %i\n", field->rows);
-    fprintf(stdout, "X array pointer: %p\n", field->x_array);
-    fprintf(stdout, "Y array pointer: %p\n", field->y_array);
-    fprintf(stdout, "Min %g\n", field->min);
-    fprintf(stdout, "Max %g\n", field->max);
-    fprintf(stdout, "Sum %g\n", field->sum);
-    fprintf(stdout, "Mean %g\n", field->mean);
-    fprintf(stdout, "Nonull %i\n", field->nonull);
-    fprintf(stdout, "X array info \n");
-    N_print_array_2d_info(field->x_array);
-    fprintf(stdout, "Y array info \n");
-    N_print_array_2d_info(field->y_array);
-
-    return;
-}
-
-
-/*!
- * \brief Allocate a N_gradient_field_3d
- *
- * The field arrays are always of type DCELL_TYPE. 
- *
- * \param cols - number of cols of the 3d array from which the gradient should be calculated
- * \param rows - number of rows of the 3d array from which the gradient should be calculated
- * \param depths - number of depths of the 3d array from which the gradient should be calculated
- * \return N_gradient_field_3d *
- *
- * */
-N_gradient_field_3d *N_alloc_gradient_field_3d(int cols, int rows, int depths)
-{
-    N_gradient_field_3d *field;
-
-    G_debug(5,
-	    "N_alloc_gradient_field_3d: allocate a N_gradient_field_3d struct");
-
-    field = (N_gradient_field_3d *) G_calloc(1, sizeof(N_gradient_field_3d));
-
-    field->x_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    field->y_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    field->z_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-
-    field->cols = cols;
-    field->rows = rows;
-    field->depths = depths;
-
-    return field;
-}
-
-
-/*!
- * \brief Free's a N_gradient_neighbours_3d structure
- *
- * \return void
- *
- * */
-void N_free_gradient_field_3d(N_gradient_field_3d * field)
-{
-
-    N_free_array_3d(field->x_array);
-    N_free_array_3d(field->y_array);
-    N_free_array_3d(field->z_array);
-
-    G_free(field);
-
-    field = NULL;
-
-    return;
-}
-
-
-/*!
- * \brief Copy N_gradient_field_3d structure from source to target
- *
- * \param source - the source N_gradient_field_3d struct
- * \param target - the target N_gradient_field_3d struct
- * \return int - 1 success, 0 failure while copying
- *
- * */
-int
-N_copy_gradient_field_3d(N_gradient_field_3d * source,
-			 N_gradient_field_3d * target)
-{
-    G_debug(3, "N_copy_gradient_field_3d: copy N_gradient_field_3d");
-
-    if (!source || !target)
-	return 0;
-
-    N_copy_array_3d(source->x_array, target->x_array);
-    N_copy_array_3d(source->y_array, target->y_array);
-    N_copy_array_3d(source->z_array, target->z_array);
-
-    return 1;
-}
-
-/*! \brief Print gradient field information to stdout
- *
- * \param field N_gradient_3d_field *
- * \return void
- *
- * */
-void N_print_gradient_field_3d_info(N_gradient_field_3d * field)
-{
-
-    fprintf(stdout, "N_gradient_field_3d \n");
-    fprintf(stdout, "Cols %i\n", field->cols);
-    fprintf(stdout, "Rows: %i\n", field->rows);
-    fprintf(stdout, "Depths %i\n", field->depths);
-    fprintf(stdout, "X array pointer: %p\n", field->x_array);
-    fprintf(stdout, "Y array pointer: %p\n", field->y_array);
-    fprintf(stdout, "Z array pointer: %p\n", field->z_array);
-    fprintf(stdout, "Min %g\n", field->min);
-    fprintf(stdout, "Max %g\n", field->max);
-    fprintf(stdout, "Sum %g\n", field->sum);
-    fprintf(stdout, "Mean %g\n", field->mean);
-    fprintf(stdout, "Nonull %i\n", field->nonull);
-    fprintf(stdout, "X array info \n");
-    N_print_array_3d_info(field->x_array);
-    fprintf(stdout, "Y array info \n");
-    N_print_array_3d_info(field->y_array);
-    fprintf(stdout, "Z array info \n");
-    N_print_array_3d_info(field->z_array);
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_gradient_calc.c
===================================================================
--- grass/trunk/lib/gpde/N_gradient_calc.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_gradient_calc.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,657 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:     	gradient management functions 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <grass/N_pde.h>
-
-/*! \brief Calculate basic statistics of a gradient field
- *
- * The statistic is stored in the gradient field struct
- *
- * \param field N_gradient_2d_field *
- * \return void
- *
- * */
-void N_calc_gradient_field_2d_stats(N_gradient_field_2d * field)
-{
-    double minx, miny;
-    double maxx, maxy;
-    double sumx, sumy;
-    int nonullx, nonully;
-
-    G_debug(3,
-	    "N_calc_gradient_field_2d_stats: compute gradient field stats");
-
-    N_calc_array_2d_stats(field->x_array, &minx, &maxx, &sumx, &nonullx, 0);
-    N_calc_array_2d_stats(field->y_array, &miny, &maxy, &sumy, &nonully, 0);
-
-    if (minx < miny)
-	field->min = minx;
-    else
-	field->min = miny;
-
-    if (maxx > maxy)
-	field->max = maxx;
-    else
-	field->max = maxy;
-
-    field->sum = sumx + sumy;
-    field->nonull = nonullx + nonully;
-    field->mean = field->sum / (double)field->nonull;
-
-    return;
-}
-
-/*!
- * \brief This function computes the gradient based on the input N_array_2d pot
- * (potential), a weighting factor N_array_2d named weight and the distance between two cells 
- * saved in the N_geom_data struct.
- *
- * The gradient is calculated between cells for each cell and direction.
- * An existing gradient field can be filled with new data or, if a NULL pointer is
- * given, a new gradient field will be allocated with the appropriate size.
- *
- *
- \verbatim
- ______________ 
- |    |    |    |
- |    |    |    |
- |----|-NC-|----|
- |    |    |    |
- |   WC    EC   |
- |    |    |    |
- |----|-SC-|----|
- |    |    |    |
- |____|____|____|
-
-
- x - direction:
-
- r = 2 * weight[row][col]*weight[row][col + 1] / (weight[row][col]*weight[row][col + 1])
- EC = r * (pot[row][col] - pot[row][col + 1])/dx
-
- y - direction:
-
- r = 2 * weight[row][col]*weight[row + 1][col] / (weight[row][col]*weight[row + 1][col])
- SC = r * (pot[row][col] - pot[row + 1][col])/dy
-
- the values SC and EC are the values of the next row/col
-
-
- \endverbatim
- * \param pot N_array_2d * - the potential N_array_2d 
- * \param weight_x N_array_2d * - the weighting factor N_array_2d used to modify the gradient in x-direction
- * \param weight_y N_array_2d * - the weighting factor N_array_2d used to modify the gradient in y-direction
- * \param geom N_geom_data * - geometry data structure
- * \param gradfield N_gradient_field_2d * - a gradient field of the correct size, if a NULL pointer is provided this gradient field will be new allocated
- * \return N_gradient_field_2d * - the pointer to the computed gradient field
-
- *
- * */
-N_gradient_field_2d *N_compute_gradient_field_2d(N_array_2d * pot,
-						 N_array_2d * weight_x,
-						 N_array_2d * weight_y,
-						 N_geom_data * geom,
-						 N_gradient_field_2d *
-						 gradfield)
-{
-    int i, j;
-    int rows, cols;
-    double dx, dy, p1, p2, r1, r2, mean, grad, res;
-    N_gradient_field_2d *field = gradfield;
-
-
-    if (pot->cols != weight_x->cols || pot->cols != weight_y->cols)
-	G_fatal_error
-	    ("N_compute_gradient_field_2d: the arrays are not of equal size");
-
-    if (pot->rows != weight_x->rows || pot->rows != weight_y->rows)
-	G_fatal_error
-	    ("N_compute_gradient_field_2d: the arrays are not of equal size");
-
-    if (pot->cols != geom->cols || pot->rows != geom->rows)
-	G_fatal_error
-	    ("N_compute_gradient_field_2d: array sizes and geometry data are different");
-
-
-    G_debug(3, "N_compute_gradient_field_2d: compute gradient field");
-
-    rows = pot->rows;
-    cols = pot->cols;
-    dx = geom->dx;
-    dy = geom->dy;
-
-    if (field == NULL) {
-	field = N_alloc_gradient_field_2d(cols, rows);
-    }
-    else {
-	if (field->cols != geom->cols || field->rows != geom->rows)
-	    G_fatal_error
-		("N_compute_gradient_field_2d: gradient field sizes and geometry data are different");
-    }
-
-
-    for (j = 0; j < rows; j++)
-	for (i = 0; i < cols - 1; i++) {
-	    grad = 0;
-	    mean = 0;
-
-	    /* Only compute if the arrays are not null */
-	    if (!N_is_array_2d_value_null(pot, i, j) &&
-		!N_is_array_2d_value_null(pot, i + 1, j)) {
-		p1 = N_get_array_2d_d_value(pot, i, j);
-		p2 = N_get_array_2d_d_value(pot, i + 1, j);
-		grad = (p1 - p2) / dx;	/* gradient */
-	    }
-	    if (!N_is_array_2d_value_null(weight_x, i, j) &&
-		!N_is_array_2d_value_null(weight_x, i + 1, j)) {
-		r1 = N_get_array_2d_d_value(weight_x, i, j);
-		r2 = N_get_array_2d_d_value(weight_x, i + 1, j);
-		mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
-	    }
-
-	    res = mean * grad;
-
-	    N_put_array_2d_d_value(field->x_array, i + 1, j, res);
-
-	}
-
-    for (j = 0; j < rows - 1; j++)
-	for (i = 0; i < cols; i++) {
-	    grad = 0;
-	    mean = 0;
-
-	    /* Only compute if the arrays are not null */
-	    if (!N_is_array_2d_value_null(pot, i, j) &&
-		!N_is_array_2d_value_null(pot, i, j + 1)) {
-		p1 = N_get_array_2d_d_value(pot, i, j);
-		p2 = N_get_array_2d_d_value(pot, i, j + 1);
-		grad = (p1 - p2) / dy;	/* gradient */
-	    }
-	    if (!N_is_array_2d_value_null(weight_y, i, j) &&
-		!N_is_array_2d_value_null(weight_y, i, j + 1)) {
-		r1 = N_get_array_2d_d_value(weight_y, i, j);
-		r2 = N_get_array_2d_d_value(weight_y, i, j + 1);
-		mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
-	    }
-
-	    res = -1 * mean * grad;
-
-	    N_put_array_2d_d_value(field->y_array, i, j + 1, res);
-
-	}
-
-    /*Compute gradient field statistics */
-    N_calc_gradient_field_2d_stats(field);
-
-    return field;
-}
-
-/*! 
- * \brief Calculate the x and y vector components from a gradient field for each 
- * cell and stores them in the provided N_array_2d structures
- *
- * The arrays must have the same size as the gradient field.
-
- \verbatim
-
- Based on this storages scheme the gradient vector for each cell is 
- calculated and stored in the provided  N_array_2d structures
-
- ______________ 
- |    |    |    |
- |    |    |    |
- |----|-NC-|----|
- |    |    |    |
- |   WC    EC   |
- |    |    |    |
- |----|-SC-|----|
- |    |    |    |
- |____|____|____|
-
- x vector component:
-
- x = (WC + EC) / 2
-
- y vector component:
-
- y = (NC + SC) / 2
-
- \endverbatim
- *
- * \param field N_gradient_field_2d *
- * \param x_comp N_array_2d * - the array in which the x component will be written
- * \param y_comp N_array_2d * - the array in which the y component will be written
- *
- * \return void
- * */
-void
-N_compute_gradient_field_components_2d(N_gradient_field_2d * field,
-				       N_array_2d * x_comp,
-				       N_array_2d * y_comp)
-{
-    int i, j;
-
-    int rows, cols;
-
-    double vx, vy;
-
-    N_array_2d *x = x_comp;
-
-    N_array_2d *y = y_comp;
-
-    N_gradient_2d grad;
-
-
-    if (!x)
-	G_fatal_error("N_compute_gradient_components_2d: x array is empty");
-    if (!y)
-	G_fatal_error("N_compute_gradient_components_2d: y array is empty");
-
-    cols = field->x_array->cols;
-    rows = field->x_array->rows;
-
-    /*Check the array sizes */
-    if (x->cols != cols || x->rows != rows)
-	G_fatal_error
-	    ("N_compute_gradient_components_2d: the size of the x array doesn't fit the gradient field size");
-    if (y->cols != cols || y->rows != rows)
-	G_fatal_error
-	    ("N_compute_gradient_components_2d: the size of the y array doesn't fit the gradient field size");
-
-    for (j = 0; j < rows; j++)
-	for (i = 0; i < cols; i++) {
-	    N_get_gradient_2d(field, &grad, i, j);
-
-	    /* in case a gradient is zero, we expect a no flow boundary */
-	    if (grad.WC == 0.0 || grad.EC == 0.0)
-		vx = (grad.WC + grad.EC);
-	    else
-		vx = (grad.WC + grad.EC) / 2;
-	    if (grad.NC == 0.0 || grad.SC == 0.0)
-		vy = (grad.NC + grad.SC);
-	    else
-		vy = (grad.NC + grad.SC) / 2;
-
-	    N_put_array_2d_d_value(x, i, j, vx);
-	    N_put_array_2d_d_value(y, i, j, vy);
-	}
-
-    return;
-}
-
-/*! \brief Calculate basic statistics of a gradient field
- *
- * The statistic is stored in the gradient field struct
- *
- * \param field N_gradient_3d_field *
- * \return void
- *
- * */
-void N_calc_gradient_field_3d_stats(N_gradient_field_3d * field)
-{
-    double minx, miny, minz;
-
-    double maxx, maxy, maxz;
-
-    double sumx, sumy, sumz;
-
-    int nonullx, nonully, nonullz;
-
-    G_debug(3,
-	    "N_calc_gradient_field_3d_stats: compute gradient field stats");
-
-    N_calc_array_3d_stats(field->x_array, &minx, &maxx, &sumx, &nonullx, 0);
-    N_calc_array_3d_stats(field->y_array, &miny, &maxy, &sumy, &nonully, 0);
-    N_calc_array_3d_stats(field->z_array, &minz, &maxz, &sumz, &nonullz, 0);
-
-    if (minx <= minz && minx <= miny)
-	field->min = minx;
-    if (miny <= minz && miny <= minx)
-	field->min = miny;
-    if (minz <= minx && minz <= miny)
-	field->min = minz;
-
-    if (maxx >= maxz && maxx >= maxy)
-	field->max = maxx;
-    if (maxy >= maxz && maxy >= maxx)
-	field->max = maxy;
-    if (maxz >= maxx && maxz >= maxy)
-	field->max = maxz;
-
-    field->sum = sumx + sumy + sumz;
-    field->nonull = nonullx + nonully + nonullz;
-    field->mean = field->sum / (double)field->nonull;
-
-    return;
-}
-
-
-/*!
- * \brief This function computes the gradient based on the input N_array_3d pot
- * (that means potential), a weighting factor N_array_3d named weight and the distance between two cells 
- * saved in the N_geom_data struct.
- *
- * The gradient is calculated between cells for each cell and direction.
- * An existing gradient field can be filled with new data or, if a NULL pointer is
- * given, a new gradient field will be allocated with the appropriate size.
- *
- *
- *
- *
- \verbatim
-
- |  /
- TC NC
- |/
- --WC-----EC--
- /|
- SC BC
- /  |
-
- x - direction:
-
- r = 2 * weight_x[depth][row][col]*weight_x[depth][row][col + 1] / (weight_X[depth][row][col]*weight_x[depth][row][col + 1])
- EC = r * (pot[depth][row][col] - pot[depth][row][col + 1])/dx
-
- y - direction:
-
- r = 2 * weight_y[depth][row][col]*weight_y[depth][row + 1][col] / (weight_y[depth][row][col]*weight_y[depth][row + 1][col])
- SC = r * (pot[depth][row][col] - pot[depth][row + 1][col])/dy
-
- z - direction:
-
- r = 2 * weight_z[depth][row][col]*weight_z[depth + 1][row][col] / (weight_z[depth][row][col]*weight_z[depth + 1][row][col])
- TC = r * (pot[depth][row][col] - pot[depth + 1][row][col])/dy
-
- the values BC, NC, WC are the values of the next depth/row/col
-
-
- \endverbatim
- * \param pot N_array_3d * - the potential N_array_2d 
- * \param weight_x N_array_3d * - the weighting factor N_array_3d used to modify the gradient in x-direction
- * \param weight_y N_array_3d * - the weighting factor N_array_3d used to modify the gradient in y-direction
- * \param weight_z N_array_3d * - the weighting factor N_array_3d used to modify the gradient in z-direction
- * \param geom N_geom_data * - geometry data structure
- * \param gradfield N_gradient_field_3d * - a gradient field of the correct size, if a NULL pointer is provided this gradient field will be new allocated
- * \return N_gradient_field_3d * - the pointer to the computed gradient field
- *
- * */
-N_gradient_field_3d *N_compute_gradient_field_3d(N_array_3d * pot,
-						 N_array_3d * weight_x,
-						 N_array_3d * weight_y,
-						 N_array_3d * weight_z,
-						 N_geom_data * geom,
-						 N_gradient_field_3d *
-						 gradfield)
-{
-    int i, j, k;
-
-    int cols, rows, depths;
-
-    double dx, dy, dz, p1, p2, r1, r2, mean, grad, res;
-
-    N_gradient_field_3d *field = gradfield;
-
-
-    if (pot->cols != weight_x->cols || pot->cols != weight_y->cols ||
-	pot->cols != weight_z->cols)
-	G_fatal_error
-	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
-
-    if (pot->rows != weight_x->rows || pot->rows != weight_y->rows ||
-	pot->rows != weight_z->rows)
-	G_fatal_error
-	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
-
-    if (pot->depths != weight_x->depths || pot->depths != weight_y->depths ||
-	pot->depths != weight_z->depths)
-	G_fatal_error
-	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
-
-    if (pot->cols != geom->cols || pot->rows != geom->rows ||
-	pot->depths != geom->depths)
-	G_fatal_error
-	    ("N_compute_gradient_field_3d: array sizes and geometry data are different");
-
-    G_debug(3, "N_compute_gradient_field_3d: compute gradient field");
-
-    cols = geom->cols;
-    rows = geom->rows;
-    depths = geom->depths;
-    dx = geom->dx;
-    dy = geom->dy;
-    dz = geom->dz;
-
-    if (gradfield == NULL) {
-	field = N_alloc_gradient_field_3d(cols, rows, depths);
-    }
-    else {
-	if (field->cols != geom->cols || field->rows != geom->rows ||
-	    field->depths != geom->depths)
-	    G_fatal_error
-		("N_compute_gradient_field_3d: gradient field sizes and geometry data are different");
-    }
-
-    for (k = 0; k < depths; k++)
-	for (j = 0; j < rows; j++)
-	    for (i = 0; i < cols - 1; i++) {
-		grad = 0;
-		mean = 0;
-
-		/*Only compute if the arrays are not null */
-		if (!N_is_array_3d_value_null(pot, i, j, k) &&
-		    !N_is_array_3d_value_null(pot, i + 1, j, k)) {
-		    p1 = N_get_array_3d_d_value(pot, i, j, k);
-		    p2 = N_get_array_3d_d_value(pot, i + 1, j, k);
-		    grad = (p1 - p2) / dx;	/* gradient */
-		}
-		if (!N_is_array_3d_value_null(weight_x, i, j, k) &&
-		    !N_is_array_3d_value_null(weight_x, i + 1, j, k)) {
-		    r1 = N_get_array_3d_d_value(weight_x, i, j, k);
-		    r2 = N_get_array_3d_d_value(weight_x, i + 1, j, k);
-		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
-		}
-
-		res = mean * grad;
-
-		G_debug(6,
-			"N_compute_gradient_field_3d: X-direction insert value %6.5g at %i %i %i ",
-			res, k, j, i + 1);
-
-		N_put_array_3d_d_value(field->x_array, i + 1, j, k, res);
-
-	    }
-
-    for (k = 0; k < depths; k++)
-	for (j = 0; j < rows - 1; j++)
-	    for (i = 0; i < cols; i++) {
-		grad = 0;
-		mean = 0;
-
-		/* Only compute if the arrays are not null */
-		if (!N_is_array_3d_value_null(pot, i, j, k) &&
-		    !N_is_array_3d_value_null(pot, i, j + 1, k)) {
-		    p1 = N_get_array_3d_d_value(pot, i, j, k);
-		    p2 = N_get_array_3d_d_value(pot, i, j + 1, k);
-		    grad = (p1 - p2) / dy;	/* gradient */
-		}
-		if (!N_is_array_3d_value_null(weight_y, i, j, k) &&
-		    !N_is_array_3d_value_null(weight_y, i, j + 1, k)) {
-		    r1 = N_get_array_3d_d_value(weight_y, i, j, k);
-		    r2 = N_get_array_3d_d_value(weight_y, i, j + 1, k);
-		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
-		}
-
-		res = -1 * mean * grad;	/*invert the direction, because we count from north to south,
-					 * but the gradient is defined in y direction */
-
-		G_debug(6,
-			"N_compute_gradient_field_3d: Y-direction insert value %6.5g at %i %i %i ",
-			res, k, j + 1, i);
-
-		N_put_array_3d_d_value(field->y_array, i, j + 1, k, res);
-
-	    }
-
-    for (k = 0; k < depths - 1; k++)
-	for (j = 0; j < rows; j++)
-	    for (i = 0; i < cols; i++) {
-		grad = 0;
-		mean = 0;
-
-		/* Only compute if the arrays are not null */
-		if (!N_is_array_3d_value_null(pot, i, j, k) &&
-		    !N_is_array_3d_value_null(pot, i, j, k + 1)) {
-		    p1 = N_get_array_3d_d_value(pot, i, j, k);
-		    p2 = N_get_array_3d_d_value(pot, i, j, k + 1);
-		    grad = (p1 - p2) / dz;	/* gradient */
-		}
-		if (!N_is_array_3d_value_null(weight_z, i, j, k) &&
-		    !N_is_array_3d_value_null(weight_z, i, j, k + 1)) {
-		    r1 = N_get_array_3d_d_value(weight_z, i, j, k);
-		    r2 = N_get_array_3d_d_value(weight_z, i, j, k + 1);
-		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
-		}
-
-		res = mean * grad;
-
-		G_debug(6,
-			"N_compute_gradient_field_3d: Z-direction insert value %6.5g at %i %i %i ",
-			res, k + 1, j, i);
-
-		N_put_array_3d_d_value(field->z_array, i, j, k + 1, res);
-
-	    }
-
-    /*Compute gradient field statistics */
-    N_calc_gradient_field_3d_stats(field);
-
-    return field;
-}
-
-/*! 
- * \brief Calculate the x, y and z vector components from a gradient field for each cell 
- *  and store them in the provided N_array_3d structures
- *
- * The arrays must have the same size as the gradient field.
- *
- \verbatim
-
- Based on this storages scheme the gradient vector for each cell is 
- calculated and stored in the provided  N_array_3d structures
-
-
- |  /
- TC NC
- |/
- --WC-----EC--
- /|
- SC BC
- /  |
-
-
- x vector component:
-
- x = (WC + EC) / 2
-
- y vector component:
-
- y = (NC + SC) / 2
-
- z vector component:
-
- z = (TC + BC) / 2
-
- \endverbatim
-
- * \param field N_gradient_field_3d *
- * \param x_comp N_array_3d * - the array in which the x component will be written
- * \param y_comp N_array_3d * - the array in which the y component will be written
- * \param z_comp N_array_3d * - the array in which the z component will be written
- *
- * \return void
- * */
-void
-N_compute_gradient_field_components_3d(N_gradient_field_3d * field,
-				       N_array_3d * x_comp,
-				       N_array_3d * y_comp,
-				       N_array_3d * z_comp)
-{
-    int i, j, k;
-
-    int rows, cols, depths;
-
-    double vx, vy, vz;
-
-    N_array_3d *x = x_comp;
-
-    N_array_3d *y = y_comp;
-
-    N_array_3d *z = z_comp;
-
-    N_gradient_3d grad;
-
-
-    if (!x)
-	G_fatal_error("N_compute_gradient_components_3d: x array is empty");
-    if (!y)
-	G_fatal_error("N_compute_gradient_components_3d: y array is empty");
-    if (!z)
-	G_fatal_error("N_compute_gradient_components_3d: z array is empty");
-
-    cols = field->x_array->cols;
-    rows = field->x_array->rows;
-    depths = field->x_array->depths;
-
-    /*Check the array sizes */
-    if (x->cols != cols || x->rows != rows || x->depths != depths)
-	G_fatal_error
-	    ("N_compute_gradient_components_3d: the size of the x array doesn't fit the gradient field size");
-    if (y->cols != cols || y->rows != rows || y->depths != depths)
-	G_fatal_error
-	    ("N_compute_gradient_components_3d: the size of the y array doesn't fit the gradient field size");
-    if (z->cols != cols || z->rows != rows || z->depths != depths)
-	G_fatal_error
-	    ("N_compute_gradient_components_3d: the size of the z array doesn't fit the gradient field size");
-
-    for (k = 0; k < depths; k++)
-	for (j = 0; j < rows; j++)
-	    for (i = 0; i < cols; i++) {
-		N_get_gradient_3d(field, &grad, i, j, k);
-		/* in case a gradient is zero, we expect a no flow boundary */
-		if (grad.WC == 0.0 || grad.EC == 0.0)
-		    vx = (grad.WC + grad.EC);
-		else
-		    vx = (grad.WC + grad.EC) / 2;
-		if (grad.NC == 0.0 || grad.SC == 0.0)
-		    vy = (grad.NC + grad.SC);
-		else
-		    vy = (grad.NC + grad.SC) / 2;
-		if (grad.TC == 0.0 || grad.BC == 0.0)
-		    vz = (grad.TC + grad.BC);
-		else
-		    vz = (grad.TC + grad.BC) / 2;
-
-		N_put_array_3d_d_value(x, i, j, k, vx);
-		N_put_array_3d_d_value(y, i, j, k, vy);
-		N_put_array_3d_d_value(z, i, j, k, vz);
-	    }
-
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_gwflow.c
===================================================================
--- grass/trunk/lib/gpde/N_gwflow.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_gwflow.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,720 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      groundwater flow in porous media 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <grass/N_gwflow.h>
-
-/* *************************************************************** */
-/* ***************** N_gwflow_data3d ***************************** */
-/* *************************************************************** */
-/*!
- * \brief Alllocate memory for the groundwater calculation data structure in 3 dimensions
- *
- * The groundwater calculation data structure will be allocated including
- * all appendant 3d and 2d arrays. The offset for the 3d arrays is one
- * to establish homogeneous Neumann boundary conditions at the calculation area border.
- * This data structure is used to create a linear equation system based on the computation of
- * groundwater flow in porous media with the finite volume method.
- *
- * \param cols   int
- * \param rows   int
- * \param depths int
- * \return N_gwflow_data3d *
- * */
-N_gwflow_data3d *N_alloc_gwflow_data3d(int cols, int rows, int depths,
-				       int river, int drain)
-{
-    N_gwflow_data3d *data;
-
-    data = (N_gwflow_data3d *) G_calloc(1, sizeof(N_gwflow_data3d));
-
-    data->phead = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->phead_start = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->status = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->hc_x = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->hc_y = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->hc_z = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->q = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->s = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->nf = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->r = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-
-    if (river) {
-	data->river_head =
-	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-	data->river_leak =
-	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-	data->river_bed = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    }
-    else {
-	data->river_head = NULL;
-	data->river_leak = NULL;
-	data->river_bed = NULL;
-    }
-
-    if (drain) {
-	data->drain_leak =
-	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-	data->drain_bed = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    }
-    else {
-	data->drain_leak = NULL;
-	data->drain_bed = NULL;
-    }
-
-    return data;
-}
-
-/* *************************************************************** */
-/* ********************* N_free_gwflow_data3d ******************** */
-/* *************************************************************** */
-/*!
- * \brief Release the memory of the groundwater flow data structure in three dimensions
- *
- * \param data N_gwflow_data3d *
- * \return void *
- * */
-
-void N_free_gwflow_data3d(N_gwflow_data3d * data)
-{
-    if (data->phead)
-	N_free_array_3d(data->phead);
-    if (data->phead_start)
-	N_free_array_3d(data->phead_start);
-    if (data->status)
-	N_free_array_3d(data->status);
-    if (data->hc_x)
-	N_free_array_3d(data->hc_x);
-    if (data->hc_y)
-	N_free_array_3d(data->hc_y);
-    if (data->hc_z)
-	N_free_array_3d(data->hc_z);
-    if (data->q)
-	N_free_array_3d(data->q);
-    if (data->s)
-	N_free_array_3d(data->s);
-    if (data->nf)
-	N_free_array_3d(data->nf);
-    if (data->r)
-	N_free_array_2d(data->r);
-    if (data->river_head)
-	N_free_array_3d(data->river_head);
-    if (data->river_leak)
-	N_free_array_3d(data->river_leak);
-    if (data->river_bed)
-	N_free_array_3d(data->river_bed);
-    if (data->drain_leak)
-	N_free_array_3d(data->drain_leak);
-    if (data->drain_bed)
-	N_free_array_3d(data->drain_bed);
-
-    G_free(data);
-
-    data = NULL;
-
-    return;
-}
-
-/* *************************************************************** */
-/* ******************** N_alloc_gwflow_data2d ******************** */
-/* *************************************************************** */
-/*!
- * \brief Alllocate memory for the groundwater calculation data structure in 2 dimensions
- * 
- * The groundwater calculation data structure will be allocated including
- * all appendant 2d arrays. The offset for the 3d arrays is one
- * to establish homogeneous Neumann boundary conditions at the calculation area border.
- * This data structure is used to create a linear equation system based on the computation of
- * groundwater flow in porous media with the finite volume method.
- *
- * \param cols int
- * \param rows int
- * \return N_gwflow_data2d *
- * */
-N_gwflow_data2d *N_alloc_gwflow_data2d(int cols, int rows, int river,
-				       int drain)
-{
-    N_gwflow_data2d *data;
-
-    data = (N_gwflow_data2d *) G_calloc(1, sizeof(N_gwflow_data2d));
-
-    data->phead = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->phead_start = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->status = N_alloc_array_2d(cols, rows, 1, CELL_TYPE);
-    data->hc_x = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->hc_y = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->q = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->s = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->nf = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->r = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->top = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->bottom = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-
-    if (river) {
-	data->river_head = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-	data->river_leak = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-	data->river_bed = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    }
-    else {
-	data->river_head = NULL;
-	data->river_leak = NULL;
-	data->river_bed = NULL;
-    }
-
-    if (drain) {
-	data->drain_leak = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-	data->drain_bed = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    }
-    else {
-	data->drain_leak = NULL;
-	data->drain_bed = NULL;
-    }
-
-
-    return data;
-}
-
-/* *************************************************************** */
-/* ****************** N_free_gwflow_data2d *********************** */
-/* *************************************************************** */
-/*!
- * \brief Release the memory of the groundwater flow data structure in two dimensions
- *
- * \param data N_gwflow_data2d *
- * \return void
- * */
-void N_free_gwflow_data2d(N_gwflow_data2d * data)
-{
-    if (data->phead)
-	N_free_array_2d(data->phead);
-    if (data->phead_start)
-	N_free_array_2d(data->phead_start);
-    if (data->status)
-	N_free_array_2d(data->status);
-    if (data->hc_x)
-	N_free_array_2d(data->hc_x);
-    if (data->hc_y)
-	N_free_array_2d(data->hc_y);
-    if (data->q)
-	N_free_array_2d(data->q);
-    if (data->s)
-	N_free_array_2d(data->s);
-    if (data->nf)
-	N_free_array_2d(data->nf);
-    if (data->r)
-	N_free_array_2d(data->r);
-    if (data->top)
-	N_free_array_2d(data->top);
-    if (data->bottom)
-	N_free_array_2d(data->bottom);
-    if (data->river_head)
-	N_free_array_2d(data->river_head);
-    if (data->river_leak)
-	N_free_array_2d(data->river_leak);
-    if (data->river_bed)
-	N_free_array_2d(data->river_bed);
-    if (data->drain_leak)
-	N_free_array_2d(data->drain_leak);
-    if (data->drain_bed)
-	N_free_array_2d(data->drain_bed);
-
-    G_free(data);
-
-    data = NULL;;
-
-    return;
-}
-
-/* *************************************************************** */
-/* ***************** N_callback_gwflow_3d ************************ */
-/* *************************************************************** */
-/*!
- * \brief This callback function creates the mass balance of a 7 point star
- *
- * The mass balance is based on the common groundwater flow equation:
- *
- * \f[Ss \frac{\partial h}{\partial t} = \nabla {\bf K} \nabla h + q \f]
- *
- * This equation is discretizised with the finite volume method in three dimensions.
- *
- *
- * \param gwdata N_gwflow_data3d *
- * \param geom N_geom_data *
- * \param col   int
- * \param row   int
- * \param depth int
- * \return N_data_star *
- *
- * */
-N_data_star *N_callback_gwflow_3d(void *gwdata, N_geom_data * geom, int col,
-				  int row, int depth)
-{
-    double hc_e = 0, hc_w = 0, hc_n = 0, hc_s = 0, hc_t = 0, hc_b = 0;
-    double dx, dy, dz, Ax, Ay, Az;
-    double hc_x, hc_y, hc_z;
-    double hc_xw, hc_yn, hc_zt;
-    double hc_xe, hc_ys, hc_zb;
-    double hc_start;
-    double Ss, r, nf, q;
-    double C, W, E, N, S, T, B, V;
-    N_data_star *mat_pos;
-    N_gwflow_data3d *data;
-
-    /*cast the void pointer to the right data structure */
-    data = (N_gwflow_data3d *) gwdata;
-
-    dx = geom->dx;
-    dy = geom->dy;
-    dz = geom->dz;
-    Az = N_get_geom_data_area_of_cell(geom, row);
-    Ay = geom->dx * geom->dz;
-    Ax = geom->dz * geom->dy;
-
-    /*read the data from the arrays */
-    hc_start = N_get_array_3d_d_value(data->phead_start, col, row, depth);
-
-    hc_x = N_get_array_3d_d_value(data->hc_x, col, row, depth);
-    hc_y = N_get_array_3d_d_value(data->hc_y, col, row, depth);
-    hc_z = N_get_array_3d_d_value(data->hc_z, col, row, depth);
-
-    hc_xw = N_get_array_3d_d_value(data->hc_x, col - 1, row, depth);
-    hc_xe = N_get_array_3d_d_value(data->hc_x, col + 1, row, depth);
-    hc_yn = N_get_array_3d_d_value(data->hc_y, col, row - 1, depth);
-    hc_ys = N_get_array_3d_d_value(data->hc_y, col, row + 1, depth);
-    hc_zt = N_get_array_3d_d_value(data->hc_z, col, row, depth + 1);
-    hc_zb = N_get_array_3d_d_value(data->hc_z, col, row, depth - 1);
-
-    hc_w = N_calc_harmonic_mean(hc_xw, hc_x);
-    hc_e = N_calc_harmonic_mean(hc_xe, hc_x);
-    hc_n = N_calc_harmonic_mean(hc_yn, hc_y);
-    hc_s = N_calc_harmonic_mean(hc_ys, hc_y);
-    hc_t = N_calc_harmonic_mean(hc_zt, hc_z);
-    hc_b = N_calc_harmonic_mean(hc_zb, hc_z);
-
-    /*inner sources */
-    q = N_get_array_3d_d_value(data->q, col, row, depth);
-    /*storativity */
-    Ss = N_get_array_3d_d_value(data->s, col, row, depth);
-    /*porosity */
-    nf = N_get_array_3d_d_value(data->nf, col, row, depth);
-
-    /*mass balance center cell to western cell */
-    W = -1 * Ax * hc_w / dx;
-    /*mass balance center cell to eastern cell */
-    E = -1 * Ax * hc_e / dx;
-    /*mass balance center cell to northern cell */
-    N = -1 * Ay * hc_n / dy;
-    /*mass balance center cell to southern cell */
-    S = -1 * Ay * hc_s / dy;
-    /*mass balance center cell to top cell */
-    T = -1 * Az * hc_t / dz;
-    /*mass balance center cell to bottom cell */
-    B = -1 * Az * hc_b / dz;
-
-    /*storativity */
-    Ss = Az * dz * Ss;
-
-    /*the diagonal entry of the matrix */
-    C = -1 * (W + E + N + S + T + B - Ss / data->dt * Az);
-
-    /*the entry in the right side b of Ax = b */
-    V = (q + hc_start * Ss / data->dt * Az);
-
-    /*only the top cells will have recharge */
-    if (depth == geom->depths - 2) {
-	r = N_get_array_2d_d_value(data->r, col, row);
-	V += r * Az;
-    }
-
-    G_debug(5, "N_callback_gwflow_3d: called [%i][%i][%i]", depth, col, row);
-
-    /*create the 7 point star entries */
-    mat_pos = N_create_7star(C, W, E, N, S, T, B, V);
-
-    return mat_pos;
-}
-
-
-/* *************************************************************** */
-/* ****************** N_gwflow_3d_calc_water_budget ************** */
-/* *************************************************************** */
-/*!
- * \brief This function computes the water budget of the entire groundwater
- *
- * The water budget is calculated for each active and dirichlet cell from
- * its surrounding neighbours. This is based on the 7 star mass balance computation
- * of N_callback_gwflow_3d and the gradient of the water heights in the cells.
- * The sum of the water budget of each active/dirichlet cell must be near zero
- * due the effect of numerical inaccuracy of cpu's.
- *
- * \param gwdata N_gwflow_data3d *
- * \param geom N_geom_data *
- * \param budget N_array_3d
- * \return void
- *
- * */
-void
-N_gwflow_3d_calc_water_budget(N_gwflow_data3d * data, N_geom_data * geom, N_array_3d * budget)
-{
-    int z, y, x, stat;
-    double h, hc;
-    double val;
-    double sum;
-    N_data_star *dstar;
-
-    int rows = data->status->rows;
-    int cols = data->status->cols;
-    int depths = data->status->depths;
-    sum = 0;
-
-    for (z = 0; z < depths; z++) {
-        for (y = 0; y < rows; y++) {
-            G_percent(y, rows - 1, 10);
-            for (x = 0; x < cols; x++) {
-                stat = (int)N_get_array_3d_d_value(data->status, x, y, z);
-
-                val = 0.0;
-
-                if (stat != N_CELL_INACTIVE ) {	/*all active/dirichlet cells */
-
-                    /* Compute the flow parameter */
-                    dstar = N_callback_gwflow_3d(data, geom, x, y, z);
-                    /* Compute the gradient in each direction pointing from the center */
-                    hc = N_get_array_3d_d_value(data->phead, x, y, z);
-
-                    if((int)N_get_array_3d_d_value(data->status, x + 1, y    , z) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x + 1, y    , z);
-                        val += dstar->E * (hc - h);
-                    }
-                    if((int)N_get_array_3d_d_value(data->status, x - 1, y    , z) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x - 1, y    , z);
-                        val += dstar->W * (hc - h);
-                    }
-                    if((int)N_get_array_3d_d_value(data->status, x    , y + 1, z) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x    , y + 1, z);
-                        val += dstar->S * (hc - h);
-                    }
-                    if((int)N_get_array_3d_d_value(data->status, x    , y - 1, z) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x    , y - 1, z);
-                        val += dstar->N * (hc - h);
-                    }
-                    if((int)N_get_array_3d_d_value(data->status, x    , y    , z + 1) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x    , y    , z + 1);
-                        val += dstar->T * (hc - h);
-                    }
-                    if((int)N_get_array_3d_d_value(data->status, x    , y    , z - 1) != N_CELL_INACTIVE) {
-                        h = N_get_array_3d_d_value(data->phead,  x    , y    , z - 1);
-                        val += dstar->B * (hc - h);
-                    }
-                    sum += val;
-
-                    G_free(dstar);
-                }
-                else {
-                    Rast_set_null_value(&val, 1, DCELL_TYPE);
-                }
-                N_put_array_3d_d_value(budget, x, y, z, val);
-            }
-        }
-    }
-
-    if(fabs(sum) < 0.0000000001)
-        G_message(_("The total sum of the water budget: %g\n"), sum);
-    else
-        G_warning(_("The total sum of the water budget is significantly larger then 0: %g\n"), sum);
-
-    return;
-}
-
-
-
-/* *************************************************************** */
-/* ****************** N_callback_gwflow_2d *********************** */
-/* *************************************************************** */
-/*!
- * \brief This callback function creates the mass balance of a 5 point star
- *
- * The mass balance is based on the common groundwater flow equation:
- *
- * \f[Ss \frac{\partial h}{\partial t} = \nabla {\bf K} \nabla h + q \f]
- *
- * This equation is discretizised with the finite volume method in two dimensions.
- *
- * \param gwdata N_gwflow_data2d *
- * \param geom N_geom_data *
- * \param col int
- * \param row int
- * \return N_data_star *
- *
- * */
-N_data_star *N_callback_gwflow_2d(void *gwdata, N_geom_data * geom, int col,
-				  int row)
-{
-    double T_e = 0, T_w = 0, T_n = 0, T_s = 0;
-    double z_e = 0, z_w = 0, z_n = 0, z_s = 0;
-    double dx, dy, Az;
-    double hc_x, hc_y;
-    double z, top;
-    double hc_xw, hc_yn;
-    double z_xw, z_yn;
-    double hc_xe, hc_ys;
-    double z_xe, z_ys;
-    double hc, hc_start;
-    double Ss, r, q;
-    double C, W, E, N, S, V;
-    N_gwflow_data2d *data;
-    N_data_star *mat_pos;
-    double river_vect = 0;	/*entry in vector */
-    double river_mat = 0;	/*entry in matrix */
-    double drain_vect = 0;	/*entry in vector */
-    double drain_mat = 0;	/*entry in matrix */
-
-    /*cast the void pointer to the right data structure */
-    data = (N_gwflow_data2d *) gwdata;
-
-    dx = geom->dx;
-    dy = geom->dy;
-    Az = N_get_geom_data_area_of_cell(geom, row);
-
-    /*read the data from the arrays */
-    hc_start = N_get_array_2d_d_value(data->phead_start, col, row);
-    hc = N_get_array_2d_d_value(data->phead, col, row);
-    top = N_get_array_2d_d_value(data->top, col, row);
-
-    /* Inner sources */
-    q = N_get_array_2d_d_value(data->q, col, row);
-
-    /* storativity or porosity of current cell face [-]*/
-    Ss = N_get_array_2d_d_value(data->s, col, row);
-    /* recharge */
-    r = N_get_array_2d_d_value(data->r, col, row) * Az;
-
-
-    if (hc > top) {		/*If the aquifer is confined */
-	z = N_get_array_2d_d_value(data->top, col,
-				   row) -
-	    N_get_array_2d_d_value(data->bottom, col, row);
-	z_xw =
-	    N_get_array_2d_d_value(data->top, col - 1,
-				   row) -
-	    N_get_array_2d_d_value(data->bottom, col - 1, row);
-	z_xe =
-	    N_get_array_2d_d_value(data->top, col + 1,
-				   row) -
-	    N_get_array_2d_d_value(data->bottom, col + 1, row);
-	z_yn =
-	    N_get_array_2d_d_value(data->top, col,
-				   row - 1) -
-	    N_get_array_2d_d_value(data->bottom, col, row - 1);
-	z_ys =
-	    N_get_array_2d_d_value(data->top, col,
-				   row + 1) -
-	    N_get_array_2d_d_value(data->bottom, col, row + 1);
-    }
-    else {			/* the aquifer is unconfined */
-
-	/* If the aquifer is unconfied use an explicite scheme to solve
-	 * the nonlinear equation. We use the phead from the first iteration */
-	z = N_get_array_2d_d_value(data->phead, col, row) -
-	    N_get_array_2d_d_value(data->bottom, col, row);
-	z_xw = N_get_array_2d_d_value(data->phead, col - 1, row) -
-	    N_get_array_2d_d_value(data->bottom, col - 1, row);
-	z_xe = N_get_array_2d_d_value(data->phead, col + 1, row) -
-	    N_get_array_2d_d_value(data->bottom, col + 1, row);
-	z_yn = N_get_array_2d_d_value(data->phead, col, row - 1) -
-	    N_get_array_2d_d_value(data->bottom, col, row - 1);
-	z_ys = N_get_array_2d_d_value(data->phead, col, row + 1) -
-	    N_get_array_2d_d_value(data->bottom, col, row + 1);
-    }
-
-    /*geometrical mean of cell height */
-    if (z_w > 0 || z_w < 0 || z_w == 0)
-	z_w = N_calc_arith_mean(z_xw, z);
-    else
-	z_w = z;
-    if (z_e > 0 || z_e < 0 || z_e == 0)
-	z_e = N_calc_arith_mean(z_xe, z);
-    else
-	z_e = z;
-    if (z_n > 0 || z_n < 0 || z_n == 0)
-	z_n = N_calc_arith_mean(z_yn, z);
-    else
-	z_n = z;
-    if (z_s > 0 || z_s < 0 || z_s == 0)
-	z_s = N_calc_arith_mean(z_ys, z);
-    else
-	z_s = z;
-
-    /*get the surrounding permeabilities */
-    hc_x = N_get_array_2d_d_value(data->hc_x, col, row);
-    hc_y = N_get_array_2d_d_value(data->hc_y, col, row);
-    hc_xw = N_get_array_2d_d_value(data->hc_x, col - 1, row);
-    hc_xe = N_get_array_2d_d_value(data->hc_x, col + 1, row);
-    hc_yn = N_get_array_2d_d_value(data->hc_y, col, row - 1);
-    hc_ys = N_get_array_2d_d_value(data->hc_y, col, row + 1);
-
-    /* calculate the transmissivities */
-    T_w = N_calc_harmonic_mean(hc_xw, hc_x) * z_w;
-    T_e = N_calc_harmonic_mean(hc_xe, hc_x) * z_e;
-    T_n = N_calc_harmonic_mean(hc_yn, hc_y) * z_n;
-    T_s = N_calc_harmonic_mean(hc_ys, hc_y) * z_s;
-
-    /* Compute the river leakage, this is an explicit method
-     * Influent and effluent flow is computed.
-     */
-    if (data->river_leak &&
-	(N_get_array_2d_d_value(data->river_leak, col, row) != 0) &&
-            N_get_array_2d_d_value(data->river_bed, col, row) <= top) {
-        /* Groundwater surface is above the river bed*/
-	if (hc > N_get_array_2d_d_value(data->river_bed, col, row)) {
-	    river_vect = N_get_array_2d_d_value(data->river_head, col, row) *
-		N_get_array_2d_d_value(data->river_leak, col, row);
-	    river_mat = N_get_array_2d_d_value(data->river_leak, col, row);
-	} /* Groundwater surface is below the river bed */
-	else if (hc < N_get_array_2d_d_value(data->river_bed, col, row)) {
-	    river_vect = (N_get_array_2d_d_value(data->river_head, col, row) -
-			  N_get_array_2d_d_value(data->river_bed, col, row))
-		* N_get_array_2d_d_value(data->river_leak, col, row);
-	    river_mat = 0;
-	}
-    }
-
-    /* compute the drainage, this is an explicit method
-     * Drainage is only enabled, if the drain bed is lower the groundwater surface
-     */
-    if (data->drain_leak &&
-	(N_get_array_2d_d_value(data->drain_leak, col, row) != 0) &&
-            N_get_array_2d_d_value(data->drain_bed, col, row) <= top) {
-	if (hc > N_get_array_2d_d_value(data->drain_bed, col, row)) {
-	    drain_vect = N_get_array_2d_d_value(data->drain_bed, col, row) *
-		N_get_array_2d_d_value(data->drain_leak, col, row);
-	    drain_mat = N_get_array_2d_d_value(data->drain_leak, col, row);
-	}
-	else if (hc <= N_get_array_2d_d_value(data->drain_bed, col, row)) {
-	    drain_vect = 0;
-	    drain_mat = 0;
-	}
-    }
-
-    /*mass balance center cell to western cell */
-    W = -1 * T_w * dy / dx;
-    /*mass balance center cell to eastern cell */
-    E = -1 * T_e * dy / dx;
-    /*mass balance center cell to northern cell */
-    N = -1 * T_n * dx / dy;
-    /*mass balance center cell to southern cell */
-    S = -1 * T_s * dx / dy;
-
-    /*the diagonal entry of the matrix */
-    C = -1 * (W + E + N + S -  Az *Ss / data->dt - river_mat * Az -
-	      drain_mat * Az);
-
-    /*the entry in the right side b of Ax = b */
-    V = (q + hc_start * Az * Ss / data->dt) + r + river_vect * Az +
-	drain_vect * Az;
-
-    G_debug(5, "N_callback_gwflow_2d: called [%i][%i]", row, col);
-
-    /*create the 5 point star entries */
-    mat_pos = N_create_5star(C, W, E, N, S, V);
-
-    return mat_pos;
-}
-
-
-
-/* *************************************************************** */
-/* ****************** N_gwflow_2d_calc_water_budget ************** */
-/* *************************************************************** */
-/*!
- * \brief This function computes the water budget of the entire groundwater
- *
- * The water budget is calculated for each active and dirichlet cell from
- * its surrounding neighbours. This is based on the 5 star mass balance computation
- * of N_callback_gwflow_2d and the gradient of the water heights in the cells.
- * The sum of the water budget of each active/dirichlet cell must be near zero
- * due the effect of numerical inaccuracy of cpu's.
- *
- * \param gwdata N_gwflow_data2d *
- * \param geom N_geom_data *
- * \param budget N_array_2d
- * \return void
- *
- * */
-void
-N_gwflow_2d_calc_water_budget(N_gwflow_data2d * data, N_geom_data * geom, N_array_2d * budget)
-{
-    int y, x, stat;
-    double h, hc;
-    double val;
-    double sum;
-    N_data_star *dstar;
-
-    int rows = data->status->rows;
-    int cols = data->status->cols;
-
-    sum = 0;
-
-    for (y = 0; y < rows; y++) {
-	G_percent(y, rows - 1, 10);
-	for (x = 0; x < cols; x++) {
-	    stat = N_get_array_2d_c_value(data->status, x, y);
-
-            val = 0.0;
-
-	    if (stat != N_CELL_INACTIVE ) {	/*all active/dirichlet cells */
-
-                /* Compute the flow parameter */
-                dstar = N_callback_gwflow_2d(data, geom, x, y);
-                /* Compute the gradient in each direction pointing from the center */
-                hc = N_get_array_2d_d_value(data->phead, x, y);
-
-                if((int)N_get_array_2d_d_value(data->status, x + 1, y    ) != N_CELL_INACTIVE) {
-                    h = N_get_array_2d_d_value(data->phead,  x + 1, y);
-                    val += dstar->E * (hc - h);
-                }
-                if((int)N_get_array_2d_d_value(data->status, x - 1, y    ) != N_CELL_INACTIVE) {
-                    h = N_get_array_2d_d_value(data->phead,  x - 1, y);
-                    val += dstar->W * (hc - h);
-                }
-                if((int)N_get_array_2d_d_value(data->status, x    , y + 1) != N_CELL_INACTIVE) {
-                    h = N_get_array_2d_d_value(data->phead,  x    , y + 1);
-                    val += dstar->S * (hc - h);
-                }
-                if((int)N_get_array_2d_d_value(data->status, x    , y - 1) != N_CELL_INACTIVE) {
-                    h = N_get_array_2d_d_value(data->phead,  x    , y - 1);
-                    val += dstar->N * (hc - h);
-                }
-
-                sum += val;
-
-                G_free(dstar);
-	    }
-	    else {
-		Rast_set_null_value(&val, 1, DCELL_TYPE);
-	    }
-	    N_put_array_2d_d_value(budget, x, y, val);
-	}
-    }
-
-    if(fabs(sum) < 0.0000000001)
-        G_message(_("The total sum of the water budget: %g\n"), sum);
-    else
-        G_warning(_("The total sum of the water budget is significantly larger then 0: %g\n"), sum);
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_heatflow.c
===================================================================
--- grass/trunk/lib/gpde/N_heatflow.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_heatflow.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,19 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      Calculation of heatflow
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <grass/N_pde.h>

Deleted: grass/trunk/lib/gpde/N_les.c
===================================================================
--- grass/trunk/lib/gpde/N_les.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_les.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,335 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      functions to manage linear equation systems
-* 		part of the gpde library
-*               
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <stdlib.h>
-#include <grass/N_pde.h>
-#include <grass/gmath.h>
-
-
-/*!
- * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A, vector x and vector b
- *
- * This function calls #N_alloc_les_param
- *
- * \param cols int
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_nquad_les(int cols, int rows, int type)
-{
-    return N_alloc_les_param(cols, rows, type, 2);
-}
-
-/*!
- * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A and vector x
- *
- * This function calls #N_alloc_les_param
- *
- * \param cols int
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_nquad_les_Ax(int cols, int rows, int type)
-{
-    return N_alloc_les_param(cols, rows, type, 1);
-}
-
-/*!
- * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A
- *
- * This function calls #N_alloc_les_param
- *
- * \param cols int
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_nquad_les_A(int cols, int rows, int type)
-{
-    return N_alloc_les_param(cols, rows, type, 0);
-}
-
-/*!
- * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A, vector x and vector b
- *
- * This function calls #N_alloc_les_param
- *
- * \param cols int
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_nquad_les_Ax_b(int cols, int rows, int type)
-{
-    return N_alloc_les_param(cols, rows, type, 2);
-}
-
-
-
-/*!
- * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A, vector x and vector b
- *
- * This function calls #N_alloc_les_param
- *
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_les(int rows, int type)
-{
-    return N_alloc_les_param(rows, rows, type, 2);
-}
-
-/*!
- * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A and vector x
- *
- * This function calls #N_alloc_les_param
- *
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_les_Ax(int rows, int type)
-{
-    return N_alloc_les_param(rows, rows, type, 1);
-}
-
-/*!
- * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A
- *
- * This function calls #N_alloc_les_param
- *
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_les_A(int rows, int type)
-{
-    return N_alloc_les_param(rows, rows, type, 0);
-}
-
-/*!
- * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A, vector x and vector b
- *
- * This function calls #N_alloc_les_param
- *
- * \param rows int
- * \param type int
- * \return N_les *
- *
- * */
-N_les *N_alloc_les_Ax_b(int rows, int type)
-{
-    return N_alloc_les_param(rows, rows, type, 2);
-}
-
-
-/*!
- * \brief Allocate memory for a quadratic or not quadratic linear equation system
- *
- * The type of the linear equation system must be N_NORMAL_LES for
- * a regular quadratic matrix or N_SPARSE_LES for a sparse matrix
- *
- * <p>
- * In case of N_NORMAL_LES
- * 
- * A quadratic matrix of size rows*rows*sizeof(double) will allocated
- *
- * <p>
- * In case of N_SPARSE_LES
- *
- * a vector of size row will be allocated, ready to hold additional allocated sparse vectors.
- * each sparse vector may have a different size.
- *
- * Parameter parts defines which parts of the les should be allocated.
- * The number of columns and rows defines if the matrix is quadratic.
- *
- * \param cols int
- * \param rows int
- * \param type int
- * \param parts int -- 2 = A, x and b; 1 = A and x; 0 = A allocated
- * \return N_les *
- *
- * */
-N_les *N_alloc_les_param(int cols, int rows, int type, int parts)
-{
-    N_les *les;
-
-    int i;
-
-    if (type == N_SPARSE_LES)
-	G_debug(2,
-		"Allocate memory for a sparse linear equation system with %i rows\n",
-		rows);
-    else
-	G_debug(2,
-		"Allocate memory for a regular linear equation system with %i rows\n",
-		rows);
-
-    les = (N_les *) G_calloc(1, sizeof(N_les));
-
-    if (parts > 0) {
-	les->x = (double *)G_calloc(cols, sizeof(double));
-	for (i = 0; i < cols; i++)
-	    les->x[i] = 0.0;
-    }
-
-
-    if (parts > 1) {
-	les->b = (double *)G_calloc(cols, sizeof(double));
-	for (i = 0; i < cols; i++)
-	    les->b[i] = 0.0;
-    }
-
-    les->A = NULL;
-    les->Asp = NULL;
-    les->rows = rows;
-    les->cols = cols;
-    if (rows == cols)
-	les->quad = 1;
-    else
-	les->quad = 0;
-
-    if (type == N_SPARSE_LES) {
-	les->Asp = G_math_alloc_spmatrix(rows);
-	les->type = N_SPARSE_LES;
-    }
-    else {
-	les->A = G_alloc_matrix(rows, cols);
-	les->type = N_NORMAL_LES;
-    }
-
-    return les;
-}
-
-/*!
- *
- * \brief prints the linear equation system to stdout
- *
- * <p>
- * Format:
- * A*x = b
- *
- * <p>
- * Example
- \verbatim
-
- 2 1 1 1 * 2 = 0.1
- 1 2 0 0 * 3 = 0.2
- 1 0 2 0 * 3 = 0.2
- 1 0 0 2 * 2 = 0.1
-
- \endverbatim
- *
- * \param les N_les * 
- * \return void
- *  
- * */
-void N_print_les(N_les * les)
-{
-    int i, j, k, out;
-
-
-    if (les->type == N_SPARSE_LES) {
-	for (i = 0; i < les->rows; i++) {
-	    for (j = 0; j < les->cols; j++) {
-		out = 0;
-		for (k = 0; k < les->Asp[i]->cols; k++) {
-		    if (les->Asp[i]->index[k] == j) {
-			fprintf(stdout, "%4.5f ", les->Asp[i]->values[k]);
-			out = 1;
-		    }
-		}
-		if (!out)
-		    fprintf(stdout, "%4.5f ", 0.0);
-	    }
-	    if (les->x)
-		fprintf(stdout, "  *  %4.5f", les->x[i]);
-	    if (les->b)
-		fprintf(stdout, " =  %4.5f ", les->b[i]);
-
-	    fprintf(stdout, "\n");
-	}
-    }
-    else {
-
-	for (i = 0; i < les->rows; i++) {
-	    for (j = 0; j < les->cols; j++) {
-		fprintf(stdout, "%4.5f ", les->A[i][j]);
-	    }
-	    if (les->x)
-		fprintf(stdout, "  *  %4.5f", les->x[i]);
-	    if (les->b)
-		fprintf(stdout, " =  %4.5f ", les->b[i]);
-
-	    fprintf(stdout, "\n");
-	}
-
-    }
-    return;
-}
-
-/*!
- * \brief Release the memory of the linear equation system
- *
- * \param les N_les *            
- * \return void
- *
- * */
-
-void N_free_les(N_les * les)
-{
-    if (les->type == N_SPARSE_LES)
-	G_debug(2, "Releasing memory of a sparse linear equation system\n");
-    else
-	G_debug(2, "Releasing memory of a regular linear equation system\n");
-
-    if (les) {
-
-	if (les->x)
-	    G_free(les->x);
-	if (les->b)
-	    G_free(les->b);
-
-	if (les->type == N_SPARSE_LES) {
-
-	    if (les->Asp) {
-		G_math_free_spmatrix(les->Asp, les->rows);
-	    }
-	}
-	else {
-
-	    if (les->A) {
-		G_free_matrix(les->A);
-	    }
-	}
-
-	free(les);
-    }
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_les_assemble.c
===================================================================
--- grass/trunk/lib/gpde/N_les_assemble.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_les_assemble.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,1397 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      functions to assemble a linear equation system
-* 		part of the gpde library
-*               
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-
-#include <math.h>
-#include <grass/N_pde.h>
-
-/* local protos */
-static int make_les_entry_2d(int i, int j, int offset_i, int offset_j,
-			     int count, int pos, N_les * les,
-			     G_math_spvector * spvect,
-			     N_array_2d * cell_count, N_array_2d * status,
-			     N_array_2d * start_val, double entry,
-			     int cell_type);
-
-static int make_les_entry_3d(int i, int j, int k, int offset_i, int offset_j,
-			     int offset_k, int count, int pos, N_les * les,
-			     G_math_spvector * spvect,
-			     N_array_3d * cell_count, N_array_3d * status,
-			     N_array_3d * start_val, double entry,
-			     int cell_type);
-
-/* *************************************************************** * 
- * ********************** N_alloc_5star ************************** * 
- * *************************************************************** */
-/*!
- * \brief allocate a 5 point star data structure
- *
- * \return N_data_star *
- * */
-N_data_star *N_alloc_5star(void)
-{
-    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
-
-    star->type = N_5_POINT_STAR;
-    star->count = 5;
-    return star;
-}
-
-/* *************************************************************** * 
- * ********************* N_alloc_7star *************************** * 
- * *************************************************************** */
-/*!
- * \brief allocate a 7 point star data structure
- *
- * \return N_data_star *
- * */
-N_data_star *N_alloc_7star(void)
-{
-    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
-
-    star->type = N_7_POINT_STAR;
-    star->count = 7;
-    return star;
-}
-
-/* *************************************************************** * 
- * ********************* N_alloc_9star *************************** * 
- * *************************************************************** */
-/*!
- * \brief allocate a 9 point star data structure
- *
- * \return N_data_star *
- *
- * \attention The 9 point start is not yet implemented in the matrix assembling function
- *
- * */
-N_data_star *N_alloc_9star(void)
-{
-    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
-
-    star->type = N_9_POINT_STAR;
-    star->count = 9;
-    return star;
-}
-
-/* *************************************************************** * 
- * ********************* N_alloc_27star ************************** * 
- * *************************************************************** */
-/*!
- * \brief allocate a 27 point star data structure
- *
- * \return N_data_star *
- *
- * \attention The 27 point start is not yet implemented in the matrix assembling function
- *
- * */
-N_data_star *N_alloc_27star(void)
-{
-    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
-
-    star->type = N_27_POINT_STAR;
-    star->count = 27;
-    return star;
-}
-
-/* *************************************************************** * 
- * ********************** N_create_5star ************************* * 
- * *************************************************************** */
-/*!
- * \brief allocate and initialize a 5 point star data structure
- *
- * \param C double
- * \param W double
- * \param E double
- * \param N double
- * \param S double
- * \param V double
- * \return N_data_star *
- * */
-N_data_star *N_create_5star(double C, double W, double E, double N,
-			    double S, double V)
-{
-    N_data_star *star = N_alloc_5star();
-
-    star->C = C;
-    star->W = W;
-    star->E = E;
-    star->N = N;
-    star->S = S;
-
-    star->V = V;
-
-    G_debug(5, "N_create_5star:  w %g e %g n %g s %g c %g v %g\n", star->W,
-	    star->E, star->N, star->S, star->C, star->V);
-
-    return star;
-}
-
-/* *************************************************************** * 
- * ************************* N_create_7star ********************** * 
- * *************************************************************** */
-/*!
- * \brief allocate and initialize a 7 point star data structure
- *
- * \param C double
- * \param W double
- * \param E double
- * \param N double
- * \param S double
- * \param T double
- * \param B double
- * \param V double
- * \return N_data_star *
- * */
-N_data_star *N_create_7star(double C, double W, double E, double N,
-			    double S, double T, double B, double V)
-{
-    N_data_star *star = N_alloc_7star();
-
-    star->C = C;
-    star->W = W;
-    star->E = E;
-    star->N = N;
-    star->S = S;
-
-    star->T = T;
-    star->B = B;
-
-    star->V = V;
-
-    G_debug(5, "N_create_7star:  w %g e %g n %g s %g t %g b %g c %g v %g\n",
-	    star->W, star->E, star->N, star->S, star->T, star->B, star->C,
-	    star->V);
-
-    return star;
-}
-
-/* *************************************************************** * 
- * ************************ N_create_9star *********************** * 
- * *************************************************************** */
-/*!
- * \brief allocate and initialize a 9 point star data structure
- *
- * \param C  double
- * \param W  double
- * \param E  double
- * \param N  double
- * \param S  double
- * \param NW double
- * \param SW double
- * \param NE double
- * \param SE double
- * \param V  double
- * \return N_data_star *
- * */
-N_data_star *N_create_9star(double C, double W, double E, double N,
-			    double S, double NW, double SW, double NE,
-			    double SE, double V)
-{
-    N_data_star *star = N_alloc_9star();
-
-    star->C = C;
-    star->W = W;
-    star->E = E;
-    star->N = N;
-    star->S = S;
-
-    star->NW = NW;
-    star->SW = SW;
-    star->NE = NE;
-    star->SE = SE;
-
-    star->V = V;
-
-    G_debug(5,
-	    "N_create_9star:  w %g e %g n %g s %g nw %g sw %g ne %g se %g c %g v %g\n",
-	    star->W, star->E, star->N, star->S, star->NW, star->SW, star->NE,
-	    star->SE, star->C, star->V);
-
-    return star;
-}
-
-/* *************************************************************** * 
- * ************************ N_create_27star *********************** * 
- * *************************************************************** */
-/*!
- * \brief allocate and initialize a 27 point star data structure
- *
- * \param C  double
- * \param W  double
- * \param E  double
- * \param N  double
- * \param S  double
- * \param NW double
- * \param SW double
- * \param NE double
- * \param SE double
- * \param T  double
- * \param W_T  double
- * \param E_T  double
- * \param N_T  double
- * \param S_T  double
- * \param NW_T double
- * \param SW_T double
- * \param NE_T double
- * \param SE_T double
- * \param B  double
- * \param W_B  double
- * \param E_B  double
- * \param N_B  double
- * \param S_B  double
- * \param NW_B double
- * \param SW_B double
- * \param NE_B double
- * \param SE_B double
- * \param V  double
- * \return N_data_star *
- * */
-N_data_star *N_create_27star(double C, double W, double E, double N, double S,
-			     double NW, double SW, double NE, double SE,
-			     double T, double W_T, double E_T, double N_T,
-			     double S_T, double NW_T, double SW_T,
-			     double NE_T, double SE_T, double B, double W_B,
-			     double E_B, double N_B, double S_B, double NW_B,
-			     double SW_B, double NE_B, double SE_B, double V)
-{
-    N_data_star *star = N_alloc_27star();
-
-    star->C = C;
-    star->W = W;
-    star->E = E;
-    star->N = N;
-    star->S = S;
-
-    star->NW = NW;
-    star->SW = SW;
-    star->NE = NE;
-    star->SE = SE;
-
-    star->T = T;
-    star->W_T = W_T;
-    star->E_T = E_T;
-    star->N_T = N_T;
-    star->S_T = S_T;
-
-    star->NW_T = NW_T;
-    star->SW_T = SW_T;
-    star->NE_T = NE_T;
-    star->SE_T = SE_T;
-
-    star->B = B;
-    star->W_B = W_B;
-    star->E_B = E_B;
-    star->N_B = N_B;
-    star->S_B = S_B;
-
-    star->NW_B = NW_B;
-    star->SW_B = SW_B;
-    star->NE_B = NE_B;
-    star->SE_B = SE_B;
-
-    star->V = V;
-
-    G_debug(5,
-	    "N_create_27star:  w %g e %g n %g s %g nw %g sw %g ne %g se %g c %g v %g\n",
-	    star->W, star->E, star->N, star->S, star->NW, star->SW, star->NE,
-	    star->SE, star->C, star->V);
-
-    G_debug(5,
-	    "N_create_27star:  w_t %g e_t %g n_t %g s_t %g nw_t %g sw_t %g ne_t %g se_t %g t %g \n",
-	    star->W_T, star->E_T, star->N_T, star->S_T, star->NW_T,
-	    star->SW_T, star->NE_T, star->SE_T, star->T);
-
-    G_debug(5,
-	    "N_create_27star:  w_b %g e_b %g n_b %g s_b %g nw_b %g sw_b %g ne_b %g se_B %g b %g\n",
-	    star->W_B, star->E_B, star->N_B, star->S_B, star->NW_B,
-	    star->SW_B, star->NE_B, star->SE_B, star->B);
-
-
-
-    return star;
-}
-
-
-/* *************************************************************** * 
- * ****************** N_set_les_callback_3d_func ***************** * 
- * *************************************************************** */
-/*!
- * \brief Set the callback function which is called while assembling the les in 3d
- *
- * \param data N_les_callback_3d *
- * \param callback_func_3d N_data_star *
- * \return void
- * */
-void
-N_set_les_callback_3d_func(N_les_callback_3d * data,
-			   N_data_star * (*callback_func_3d) ())
-{
-    data->callback = callback_func_3d;
-}
-
-/* *************************************************************** * 
- * *************** N_set_les_callback_2d_func ******************** * 
- * *************************************************************** */
-/*!
- * \brief Set the callback function which is called while assembling the les in 2d
- *
- * \param data N_les_callback_2d *
- * \param callback_func_2d N_data_star * 
- * \return void
- * */
-void
-N_set_les_callback_2d_func(N_les_callback_2d * data,
-			   N_data_star * (*callback_func_2d) ())
-{
-    data->callback = callback_func_2d;
-}
-
-/* *************************************************************** * 
- * ************** N_alloc_les_callback_3d ************************ * 
- * *************************************************************** */
-/*!
- * \brief Allocate the structure holding the callback function
- *
- * A template callback is set. Use N_set_les_callback_3d_func
- * to set up a specific function.
- *
- * \return N_les_callback_3d *
- * */
-N_les_callback_3d *N_alloc_les_callback_3d(void)
-{
-    N_les_callback_3d *call;
-
-    call = (N_les_callback_3d *) G_calloc(1, sizeof(N_les_callback_3d *));
-    call->callback = N_callback_template_3d;
-
-    return call;
-}
-
-/* *************************************************************** * 
- * *************** N_alloc_les_callback_2d *********************** * 
- * *************************************************************** */
-/*!
- * \brief Allocate the structure holding the callback function
- *
- * A template callback is set. Use N_set_les_callback_2d_func
- * to set up a specific function.
- *
- * \return N_les_callback_2d *
- * */
-N_les_callback_2d *N_alloc_les_callback_2d(void)
-{
-    N_les_callback_2d *call;
-
-    call = (N_les_callback_2d *) G_calloc(1, sizeof(N_les_callback_2d *));
-    call->callback = N_callback_template_2d;
-
-    return call;
-}
-
-/* *************************************************************** * 
- * ******************** N_callback_template_3d ******************* * 
- * *************************************************************** */
-/*!
- * \brief A callback template creates a 7 point star structure
- *
- * This is a template callback for mass balance calculation with 7 point stars
- * based on 3d data (g3d).
- *
- * \param data void *
- * \param geom N_geom_data *
- * \param depth int
- * \param row   int
- * \param col   int
- * \return N_data_star *
- *
- * */
-N_data_star *N_callback_template_3d(void *data, N_geom_data * geom, int col,
-				    int row, int depth)
-{
-    N_data_star *star = N_alloc_7star();
-
-    star->E = 1 / geom->dx;
-    star->W = 1 / geom->dx;
-    star->N = 1 / geom->dy;
-    star->S = 1 / geom->dy;
-    star->T = 1 / geom->dz;
-    star->B = 1 / geom->dz;
-    star->C = -1 * (2 / geom->dx + 2 / geom->dy + 2 / geom->dz);
-    star->V = -1;
-
-    G_debug(5,
-	    "N_callback_template_3d:  w %g e %g n %g s %g t %g b %g c %g v %g\n",
-	    star->W, star->E, star->N, star->S, star->T, star->B, star->C,
-	    star->V);
-
-
-    return star;
-}
-
-/* *************************************************************** * 
- * ********************* N_callback_template_2d ****************** * 
- * *************************************************************** */
-/*!
- * \brief A callback template creates a 9 point star structure
- *
- * This is a template callback for mass balance calculation with 9 point stars
- * based on 2d data (raster).
- *
- * \param data void *
- * \param geom N_geom_data *
- * \param row int
- * \param col int
- * \return N_data_star *
- *
- * */
-N_data_star *N_callback_template_2d(void *data, N_geom_data * geom, int col,
-				    int row)
-{
-    N_data_star *star = N_alloc_9star();
-
-    star->E = 1 / geom->dx;
-    star->NE = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
-    star->SE = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
-    star->W = 1 / geom->dx;
-    star->NW = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
-    star->SW = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
-    star->N = 1 / geom->dy;
-    star->S = 1 / geom->dy;
-    star->C =
-	-1 * (star->E + star->NE + star->SE + star->W + star->NW + star->SW +
-	      star->N + star->S);
-    star->V = 0;
-
-    return star;
-}
-
-/* *************************************************************** * 
- * ******************** N_assemble_les_2d ************************ * 
- * *************************************************************** */
-/*!
- * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active cells
- *
- * This function calls #N_assemble_les_2d_param
- *
- */
-N_les *N_assemble_les_2d(int les_type, N_geom_data * geom,
-			 N_array_2d * status, N_array_2d * start_val,
-			 void *data, N_les_callback_2d * call)
-{
-    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_ACTIVE);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active cells
- *
- * This function calls #N_assemble_les_2d_param
- *
- */
-N_les *N_assemble_les_2d_active(int les_type, N_geom_data * geom,
-				N_array_2d * status, N_array_2d * start_val,
-				void *data, N_les_callback_2d * call)
-{
-    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_ACTIVE);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active and dirichlet cells
- *
- * This function calls #N_assemble_les_2d_param
- *
- */
-N_les *N_assemble_les_2d_dirichlet(int les_type, N_geom_data * geom,
-				   N_array_2d * status,
-				   N_array_2d * start_val, void *data,
-				   N_les_callback_2d * call)
-{
-    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_DIRICHLET);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 2d location data  (raster)
- *
- * 
- * The linear equation system type can be set to N_NORMAL_LES to create a regular
- * matrix, or to N_SPARSE_LES to create a sparse matrix. This function returns
- * a new created linear equation system which can be solved with 
- * linear equation solvers. An 2d array with start values and an 2d status array
- * must be provided as well as the location geometry and a void pointer to data 
- * passed to the callback which creates the les row entries. This callback
- * must be defined in the N_les_callback_2d strcuture.
- *
- * The creation of the les is parallelized with OpenMP. 
- * If you implement new callbacks, please make sure that the 
- * function calls are thread safe.
- *
- *
- * the les can be created in two ways, with dirichlet and similar cells and without them,
- * to spare some memory. If the les is created with dirichlet cell, the dirichlet boundary condition
- * must be added.
- *
- * \param les_type int
- * \param geom      N_geom_data*
- * \param status    N_array_2d *
- * \param start_val N_array_2d *
- * \param data void *
- * \param cell_type int  -- les assemble based on N_CELL_ACTIVE or N_CELL_DIRICHLET
- * \param call N_les_callback_2d *
- * \return N_les *
- * */
-N_les *N_assemble_les_2d_param(int les_type, N_geom_data * geom,
-			       N_array_2d * status, N_array_2d * start_val,
-			       void *data, N_les_callback_2d * call,
-			       int cell_type)
-{
-    int i, j, count = 0, pos = 0;
-    int cell_type_count = 0;
-    int **index_ij;
-    N_array_2d *cell_count;
-    N_les *les = NULL;
-
-    G_debug(2,
-	    "N_assemble_les_2d: starting to assemble the linear equation system");
-
-    /* At first count the number of valid cells and save 
-     * each number in a new 2d array. Those numbers are used 
-     * to create the linear equation system.
-     * */
-
-    cell_count = N_alloc_array_2d(geom->cols, geom->rows, 1, CELL_TYPE);
-
-    /* include dirichlet cells in the les */
-    if (cell_type == N_CELL_DIRICHLET) {
-	for (j = 0; j < geom->rows; j++) {
-	    for (i = 0; i < geom->cols; i++) {
-		/*use all non-inactive cells for les creation */
-		if (N_CELL_INACTIVE < N_get_array_2d_c_value(status, i, j) &&
-		    N_get_array_2d_c_value(status, i, j) < N_MAX_CELL_STATE)
-		    cell_type_count++;
-	    }
-	}
-    }
-    /*use only active cell in the les */
-    if (cell_type == N_CELL_ACTIVE) {
-	for (j = 0; j < geom->rows; j++) {
-	    for (i = 0; i < geom->cols; i++) {
-		/*count only active cells */
-		if (N_CELL_ACTIVE == N_get_array_2d_d_value(status, i, j))
-		    cell_type_count++;
-	    }
-	}
-    }
-
-    G_debug(2, "N_assemble_les_2d: number of used cells %i\n",
-	    cell_type_count);
-
-    if (cell_type_count == 0)
-	G_fatal_error
-	    ("Not enough cells [%i] to create the linear equation system. Check the cell status. Only active cells (value = 1) are used to create the equation system.",
-	     cell_type_count);
-
-    /* Then allocate the memory for the linear equation system (les). 
-     * Only valid cells are used to create the les. */
-    index_ij = (int **)G_calloc(cell_type_count, sizeof(int *));
-    for (i = 0; i < cell_type_count; i++)
-	index_ij[i] = (int *)G_calloc(2, sizeof(int));
-
-    les = N_alloc_les_Ax_b(cell_type_count, les_type);
-
-    count = 0;
-
-    /*count the number of cells which should be used to create the linear equation system */
-    /*save the i and j indices and create a ordered numbering */
-    for (j = 0; j < geom->rows; j++) {
-	for (i = 0; i < geom->cols; i++) {
-	    /*count every non-inactive cell */
-	    if (cell_type == N_CELL_DIRICHLET) {
-		if (N_CELL_INACTIVE < N_get_array_2d_c_value(status, i, j) &&
-		    N_get_array_2d_c_value(status, i, j) < N_MAX_CELL_STATE) {
-		    N_put_array_2d_c_value(cell_count, i, j, count);
-		    index_ij[count][0] = i;
-		    index_ij[count][1] = j;
-		    count++;
-		    G_debug(5,
-			    "N_assemble_les_2d: non-inactive cells count %i at pos x[%i] y[%i]\n",
-			    count, i, j);
-		}
-		/*count every active cell */
-	    }
-	    else if (N_CELL_ACTIVE == N_get_array_2d_c_value(status, i, j)) {
-		N_put_array_2d_c_value(cell_count, i, j, count);
-		index_ij[count][0] = i;
-		index_ij[count][1] = j;
-		count++;
-		G_debug(5,
-			"N_assemble_les_2d: active cells count %i at pos x[%i] y[%i]\n",
-			count, i, j);
-	    }
-	}
-    }
-
-    G_debug(2, "N_assemble_les_2d: starting the parallel assemble loop");
-
-    /* Assemble the matrix in parallel */
-#pragma omp parallel for private(i, j, pos, count) schedule(static)
-    for (count = 0; count < cell_type_count; count++) {
-	i = index_ij[count][0];
-	j = index_ij[count][1];
-
-	/*create the entries for the */
-	N_data_star *items = call->callback(data, geom, i, j);
-
-	/* we need a sparse vector pointer anytime */
-	G_math_spvector *spvect = NULL;
-
-	/*allocate a sprase vector */
-	if (les_type == N_SPARSE_LES) {
-	    spvect = G_math_alloc_spvector(items->count);
-	}
-	/* initial conditions */
-	les->x[count] = N_get_array_2d_d_value(start_val, i, j);
-
-	/* the entry in the vector b */
-	les->b[count] = items->V;
-
-	/* pos describes the position in the sparse vector.
-	 * the first entry is always the diagonal entry of the matrix*/
-	pos = 0;
-
-	if (les_type == N_SPARSE_LES) {
-	    spvect->index[pos] = count;
-	    spvect->values[pos] = items->C;
-	}
-	else {
-	    les->A[count][count] = items->C;
-	}
-	/* western neighbour, entry is col - 1 */
-	if (i > 0) {
-	    pos = make_les_entry_2d(i, j, -1, 0, count, pos, les, spvect,
-				    cell_count, status, start_val, items->W,
-				    cell_type);
-	}
-	/* eastern neighbour, entry col + 1 */
-	if (i < geom->cols - 1) {
-	    pos = make_les_entry_2d(i, j, 1, 0, count, pos, les, spvect,
-				    cell_count, status, start_val, items->E,
-				    cell_type);
-	}
-	/* northern neighbour, entry row - 1 */
-	if (j > 0) {
-	    pos =
-		make_les_entry_2d(i, j, 0, -1, count, pos, les, spvect,
-				  cell_count, status, start_val, items->N,
-				  cell_type);
-	}
-	/* southern neighbour, entry row + 1 */
-	if (j < geom->rows - 1) {
-	    pos = make_les_entry_2d(i, j, 0, 1, count, pos, les, spvect,
-				    cell_count, status, start_val, items->S,
-				    cell_type);
-	}
-	/*in case of a nine point star, we have additional entries */
-	if (items->type == N_9_POINT_STAR) {
-	    /* north-western neighbour, entry is col - 1 row - 1 */
-	    if (i > 0 && j > 0) {
-		pos = make_les_entry_2d(i, j, -1, -1, count, pos, les, spvect,
-					cell_count, status, start_val,
-					items->NW, cell_type);
-	    }
-	    /* north-eastern neighbour, entry col + 1 row - 1 */
-	    if (i < geom->cols - 1 && j > 0) {
-		pos = make_les_entry_2d(i, j, 1, -1, count, pos, les, spvect,
-					cell_count, status, start_val,
-					items->NE, cell_type);
-	    }
-	    /* south-western neighbour, entry is col - 1 row + 1 */
-	    if (i > 0 && j < geom->rows - 1) {
-		pos = make_les_entry_2d(i, j, -1, 1, count, pos, les, spvect,
-					cell_count, status, start_val,
-					items->SW, cell_type);
-	    }
-	    /* south-eastern neighbour, entry col + 1 row + 1 */
-	    if (i < geom->cols - 1 && j < geom->rows - 1) {
-		pos = make_les_entry_2d(i, j, 1, 1, count, pos, les, spvect,
-					cell_count, status, start_val,
-					items->SE, cell_type);
-	    }
-	}
-
-	/*How many entries in the les */
-	if (les->type == N_SPARSE_LES) {
-	    spvect->cols = pos + 1;
-	    G_math_add_spvector(les->Asp, spvect, count);
-	}
-
-	if (items)
-	    G_free(items);
-    }
-
-    /*release memory */
-    N_free_array_2d(cell_count);
-
-    for (i = 0; i < cell_type_count; i++)
-	G_free(index_ij[i]);
-
-    G_free(index_ij);
-
-    return les;
-}
-
-/*!
- * \brief Integrate Dirichlet or Transmission boundary conditions into the les (2s)
- *
- * Dirichlet and Transmission boundary conditions will be integrated into
- * the provided linear equation system. This is meaningfull if
- * the les was created with #N_assemble_les_2d_dirichlet, because in
- * this case Dirichlet boundary conditions are not automatically included.
- *
- * The provided les will be modified:
- *
- * Ax = b will be splitted into Ax_u + Ax_d = b
- *
- * x_u - the unknowns
- * x_d - the Dirichlet cells
- *
- * Ax_u = b -Ax_d will be computed. Then the matrix A will be modified to
- *
- * | A_u  0 | x_u
- * |  0   I | x_d
- *
- * \param les N_les* -- the linear equation system
- * \param geom N_geom_data* -- geometrical data information
- * \param status N_array_2d* -- the status array containing the cell types
- * \param start_val N_array_2d* -- an array with start values
- * \return int -- 1 = success, 0 = failure
- * */
-int N_les_integrate_dirichlet_2d(N_les * les, N_geom_data * geom,
-				 N_array_2d * status, N_array_2d * start_val)
-{
-    int rows, cols;
-    int count = 0;
-    int i, j, x, y, stat;
-    double *dvect1;
-    double *dvect2;
-
-    G_debug(2,
-	    "N_les_integrate_dirichlet_2d: integrating the dirichlet boundary condition");
-
-    rows = geom->rows;
-    cols = geom->cols;
-
-    /*we nned to additional vectors */
-    dvect1 = (double *)G_calloc(les->cols, sizeof(double));
-    dvect2 = (double *)G_calloc(les->cols, sizeof(double));
-
-    /*fill the first one with the x vector data of Dirichlet cells */
-    count = 0;
-    for (y = 0; y < rows; y++) {
-	for (x = 0; x < cols; x++) {
-	    stat = N_get_array_2d_c_value(status, x, y);
-	    if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
-		dvect1[count] = N_get_array_2d_d_value(start_val, x, y);
-		count++;
-	    }
-	    else if (stat == N_CELL_ACTIVE) {
-		dvect1[count] = 0.0;
-		count++;
-	    }
-	}
-    }
-
-#pragma omp parallel default(shared)
-    {
-	/*perform the matrix vector product and */
-	if (les->type == N_SPARSE_LES)
-	    G_math_Ax_sparse(les->Asp, dvect1, dvect2, les->rows);
-	else
-	    G_math_d_Ax(les->A, dvect1, dvect2, les->rows, les->cols);
-#pragma omp for schedule (static) private(i)
-	for (i = 0; i < les->cols; i++)
-	    les->b[i] = les->b[i] - dvect2[i];
-    }
-
-    /*now set the Dirichlet cell rows and cols to zero and the 
-     * diagonal entry to 1*/
-    count = 0;
-    for (y = 0; y < rows; y++) {
-	for (x = 0; x < cols; x++) {
-	    stat = N_get_array_2d_c_value(status, x, y);
-	    if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
-		if (les->type == N_SPARSE_LES) {
-		    /*set the rows to zero */
-		    for (i = 0; i < les->Asp[count]->cols; i++)
-			les->Asp[count]->values[i] = 0.0;
-		    /*set the cols to zero */
-		    for (i = 0; i < les->rows; i++) {
-			for (j = 0; j < les->Asp[i]->cols; j++) {
-			    if (les->Asp[i]->index[j] == count)
-				les->Asp[i]->values[j] = 0.0;
-			}
-		    }
-
-		    /*entry on the diagonal */
-		    les->Asp[count]->values[0] = 1.0;
-
-		}
-		else {
-		    /*set the rows to zero */
-		    for (i = 0; i < les->cols; i++)
-			les->A[count][i] = 0.0;
-		    /*set the cols to zero */
-		    for (i = 0; i < les->rows; i++)
-			les->A[i][count] = 0.0;
-
-		    /*entry on the diagonal */
-		    les->A[count][count] = 1.0;
-		}
-	    }
-	    if (stat >= N_CELL_ACTIVE)
-		count++;
-	}
-    }
-
-    return 0;
-
-}
-
-/* **************************************************************** */
-/* **** make an entry in the les (2d) ***************************** */
-/* **************************************************************** */
-int make_les_entry_2d(int i, int j, int offset_i, int offset_j, int count,
-		      int pos, N_les * les, G_math_spvector * spvect,
-		      N_array_2d * cell_count, N_array_2d * status,
-		      N_array_2d * start_val, double entry, int cell_type)
-{
-    int K;
-    int di = offset_i;
-    int dj = offset_j;
-
-    K = N_get_array_2d_c_value(cell_count, i + di, j + dj) -
-	N_get_array_2d_c_value(cell_count, i, j);
-
-    /* active cells build the linear equation system */
-    if (cell_type == N_CELL_ACTIVE) {
-	/* dirichlet or transmission cells must be handled like this */
-	if (N_get_array_2d_c_value(status, i + di, j + dj) > N_CELL_ACTIVE &&
-	    N_get_array_2d_c_value(status, i + di, j + dj) < N_MAX_CELL_STATE)
-	    les->b[count] -=
-		N_get_array_2d_d_value(start_val, i + di, j + dj) * entry;
-	else if (N_get_array_2d_c_value(status, i + di, j + dj) ==
-		 N_CELL_ACTIVE) {
-	    if ((count + K) >= 0 && (count + K) < les->cols) {
-		G_debug(5,
-			" make_les_entry_2d: (N_CELL_ACTIVE) create matrix entry at row[%i] col[%i] value %g\n",
-			count, count + K, entry);
-		pos++;
-		if (les->type == N_SPARSE_LES) {
-		    spvect->index[pos] = count + K;
-		    spvect->values[pos] = entry;
-		}
-		else {
-		    les->A[count][count + K] = entry;
-		}
-	    }
-	}
-    }				/* if dirichlet cells should be used then check for all valid cell neighbours */
-    else if (cell_type == N_CELL_DIRICHLET) {
-	/* all valid cells */
-	if (N_get_array_2d_c_value(status, i + di, j + dj) > N_CELL_INACTIVE
-	    && N_get_array_2d_c_value(status, i + di,
-				      j + dj) < N_MAX_CELL_STATE) {
-	    if ((count + K) >= 0 && (count + K) < les->cols) {
-		G_debug(5,
-			" make_les_entry_2d: (N_CELL_DIRICHLET) create matrix entry at row[%i] col[%i] value %g\n",
-			count, count + K, entry);
-		pos++;
-		if (les->type == N_SPARSE_LES) {
-		    spvect->index[pos] = count + K;
-		    spvect->values[pos] = entry;
-		}
-		else {
-		    les->A[count][count + K] = entry;
-		}
-	    }
-	}
-    }
-
-    return pos;
-}
-
-
-/* *************************************************************** * 
- * ******************** N_assemble_les_3d ************************ * 
- * *************************************************************** */
-/*!
- * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active cells
- *
- * This function calls #N_assemble_les_3d_param
- * */
-N_les *N_assemble_les_3d(int les_type, N_geom_data * geom,
-			 N_array_3d * status, N_array_3d * start_val,
-			 void *data, N_les_callback_3d * call)
-{
-    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_ACTIVE);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active cells
- *
- * This function calls #N_assemble_les_3d_param
- * */
-N_les *N_assemble_les_3d_active(int les_type, N_geom_data * geom,
-				N_array_3d * status, N_array_3d * start_val,
-				void *data, N_les_callback_3d * call)
-{
-    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_ACTIVE);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active and dirichlet cells
- *
- * This function calls #N_assemble_les_3d_param
- * */
-N_les *N_assemble_les_3d_dirichlet(int les_type, N_geom_data * geom,
-				   N_array_3d * status,
-				   N_array_3d * start_val, void *data,
-				   N_les_callback_3d * call)
-{
-    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
-				   call, N_CELL_DIRICHLET);
-}
-
-/*!
- * \brief Assemble a linear equation system (les) based on 3d location data (g3d)
- *
- * The linear equation system type can be set to N_NORMAL_LES to create a regular
- * matrix, or to N_SPARSE_LES to create a sparse matrix. This function returns
- * a new created linear equation system which can be solved with 
- * linear equation solvers. An 3d array with start values and an 3d status array
- * must be provided as well as the location geometry and a void pointer to data 
- * passed to the callback which creates the les row entries. This callback
- * must be defined in the N_les_callback_3d structure.
- * 
- * The creation of the les is parallelized with OpenMP. 
- * If you implement new callbacks, please make sure that the 
- * function calls are thread safe.
- *
- * the les can be created in two ways, with dirichlet and similar cells and without them,
- * to spare some memory. If the les is created with dirichlet cell, the dirichlet boundary condition
- * must be added.
- *
- * \param les_type int
- * \param geom      N_geom_data*
- * \param status    N_array_3d *
- * \param start_val N_array_3d *
- * \param data void *
- * \param call N_les_callback_3d *
- * \param cell_type int  -- les assemble based on N_CELL_ACTIVE or N_CELL_DIRICHLET
- * \return N_les *
- * */
-N_les *N_assemble_les_3d_param(int les_type, N_geom_data * geom,
-			       N_array_3d * status, N_array_3d * start_val,
-			       void *data, N_les_callback_3d * call,
-			       int cell_type)
-{
-    int i, j, k, count = 0, pos = 0;
-    int cell_type_count = 0;
-    N_array_3d *cell_count;
-    N_les *les = NULL;
-    int **index_ij;
-
-    G_debug(2,
-	    "N_assemble_les_3d: starting to assemble the linear equation system");
-
-    cell_count =
-	N_alloc_array_3d(geom->cols, geom->rows, geom->depths, 1, DCELL_TYPE);
-
-    /* First count the number of valid cells and save  
-     * each number in a new 3d array. Those numbers are used 
-     * to create the linear equation system.*/
-
-    if (cell_type == N_CELL_DIRICHLET) {
-	/* include dirichlet cells in the les */
-	for (k = 0; k < geom->depths; k++) {
-	    for (j = 0; j < geom->rows; j++) {
-		for (i = 0; i < geom->cols; i++) {
-		    /*use all non-inactive cells for les creation */
-		    if (N_CELL_INACTIVE <
-			(int)N_get_array_3d_d_value(status, i, j, k) &&
-			(int)N_get_array_3d_d_value(status, i, j,
-						    k) < N_MAX_CELL_STATE)
-			cell_type_count++;
-		}
-	    }
-	}
-    }
-    else {
-	/*use only active cell in the les */
-	for (k = 0; k < geom->depths; k++) {
-	    for (j = 0; j < geom->rows; j++) {
-		for (i = 0; i < geom->cols; i++) {
-		    /*count only active cells */
-		    if (N_CELL_ACTIVE
-			== (int)N_get_array_3d_d_value(status, i, j, k))
-			cell_type_count++;
-
-		}
-	    }
-	}
-    }
-
-    G_debug(2,
-	    "N_assemble_les_3d: number of  used cells %i\n", cell_type_count);
-
-    if (cell_type_count == 0.0)
-	G_fatal_error
-	    ("Not enough active cells [%i] to create the linear equation system. Check the cell status. Only active cells (value = 1) are used to create the equation system.",
-	     cell_type_count);
-
-    /* allocate the memory for the linear equation system (les). 
-     * Only valid cells are used to create the les. */
-    les = N_alloc_les_Ax_b(cell_type_count, les_type);
-
-    index_ij = (int **)G_calloc(cell_type_count, sizeof(int *));
-    for (i = 0; i < cell_type_count; i++)
-	index_ij[i] = (int *)G_calloc(3, sizeof(int));
-
-    count = 0;
-    /*count the number of cells which should be used to create the linear equation system */
-    /*save the k, i and j indices and create a ordered numbering */
-    for (k = 0; k < geom->depths; k++) {
-	for (j = 0; j < geom->rows; j++) {
-	    for (i = 0; i < geom->cols; i++) {
-		if (cell_type == N_CELL_DIRICHLET) {
-		    if (N_CELL_INACTIVE <
-			(int)N_get_array_3d_d_value(status, i, j, k) &&
-			(int)N_get_array_3d_d_value(status, i, j,
-						    k) < N_MAX_CELL_STATE) {
-			N_put_array_3d_d_value(cell_count, i, j, k, count);
-			index_ij[count][0] = i;
-			index_ij[count][1] = j;
-			index_ij[count][2] = k;
-			count++;
-			G_debug(5,
-				"N_assemble_les_3d: non-inactive cells count %i at pos x[%i] y[%i] z[%i]\n",
-				count, i, j, k);
-		    }
-		}
-		else if (N_CELL_ACTIVE ==
-			 (int)N_get_array_3d_d_value(status, i, j, k)) {
-		    N_put_array_3d_d_value(cell_count, i, j, k, count);
-		    index_ij[count][0] = i;
-		    index_ij[count][1] = j;
-		    index_ij[count][2] = k;
-		    count++;
-		    G_debug(5,
-			    "N_assemble_les_3d: active cells count %i at pos x[%i] y[%i] z[%i]\n",
-			    count, i, j, k);
-		}
-	    }
-	}
-    }
-
-    G_debug(2, "N_assemble_les_3d: starting the parallel assemble loop");
-
-#pragma omp parallel for private(i, j, k, pos, count) schedule(static)
-    for (count = 0; count < cell_type_count; count++) {
-	i = index_ij[count][0];
-	j = index_ij[count][1];
-	k = index_ij[count][2];
-
-	/*create the entries for the */
-	N_data_star *items = call->callback(data, geom, i, j, k);
-
-	G_math_spvector *spvect = NULL;
-
-	/*allocate a sprase vector */
-	if (les_type == N_SPARSE_LES)
-	    spvect = G_math_alloc_spvector(items->count);
-	/* initial conditions */
-
-	les->x[count] = N_get_array_3d_d_value(start_val, i, j, k);
-
-	/* the entry in the vector b */
-	les->b[count] = items->V;
-
-	/* pos describes the position in the sparse vector.
-	 * the first entry is always the diagonal entry of the matrix*/
-	pos = 0;
-
-	if (les_type == N_SPARSE_LES) {
-	    spvect->index[pos] = count;
-	    spvect->values[pos] = items->C;
-	}
-	else {
-	    les->A[count][count] = items->C;
-	}
-	/* western neighbour, entry is col - 1 */
-	if (i > 0) {
-	    pos =
-		make_les_entry_3d(i, j, k, -1, 0, 0, count, pos, les, spvect,
-				  cell_count, status, start_val, items->W,
-				  cell_type);
-	}
-	/* eastern neighbour, entry col + 1 */
-	if (i < geom->cols - 1) {
-	    pos = make_les_entry_3d(i, j, k, 1, 0, 0, count, pos, les, spvect,
-				    cell_count, status, start_val, items->E,
-				    cell_type);
-	}
-	/* northern neighbour, entry row -1 */
-	if (j > 0) {
-	    pos =
-		make_les_entry_3d(i, j, k, 0, -1, 0, count, pos, les, spvect,
-				  cell_count, status, start_val, items->N,
-				  cell_type);
-	}
-	/* southern neighbour, entry row +1 */
-	if (j < geom->rows - 1) {
-	    pos = make_les_entry_3d(i, j, k, 0, 1, 0, count, pos, les, spvect,
-				    cell_count, status, start_val, items->S,
-				    cell_type);
-	}
-	/*only for a 7 star entry needed */
-	if (items->type == N_7_POINT_STAR || items->type == N_27_POINT_STAR) {
-	    /* the upper cell (top), entry depth + 1 */
-	    if (k < geom->depths - 1) {
-		pos =
-		    make_les_entry_3d(i, j, k, 0, 0, 1, count, pos, les,
-				      spvect, cell_count, status, start_val,
-				      items->T, cell_type);
-	    }
-	    /* the lower cell (bottom), entry depth - 1 */
-	    if (k > 0) {
-		pos =
-		    make_les_entry_3d(i, j, k, 0, 0, -1, count, pos, les,
-				      spvect, cell_count, status, start_val,
-				      items->B, cell_type);
-	    }
-	}
-
-	/*How many entries in the les */
-	if (les->type == N_SPARSE_LES) {
-	    spvect->cols = pos + 1;
-	    G_math_add_spvector(les->Asp, spvect, count);
-	}
-
-	if (items)
-	    G_free(items);
-    }
-
-    N_free_array_3d(cell_count);
-
-    for (i = 0; i < cell_type_count; i++)
-	G_free(index_ij[i]);
-
-    G_free(index_ij);
-
-    return les;
-}
-
-/*!
- * \brief Integrate Dirichlet or Transmission boundary conditions into the les (3d)
- *
- * Dirichlet and Transmission boundary conditions will be integrated into
- * the provided linear equation system. This is meaningfull if
- * the les was created with #N_assemble_les_2d_dirichlet, because in
- * this case Dirichlet boundary conditions are not automatically included.
- *
- * The provided les will be modified:
- *
- * Ax = b will be splitted into Ax_u + Ax_d = b
- *
- * x_u - the unknowns
- * x_d - the Dirichlet cells
- *
- * Ax_u = b -Ax_d will be computed. Then the matrix A will be modified to
- *
- * | A_u  0 | x_u
- * |  0   I | x_d
- *
- * \param les N_les* -- the linear equation system
- * \param geom N_geom_data* -- geometrical data information
- * \param status N_array_2d* -- the status array containing the cell types
- * \param start_val N_array_2d* -- an array with start values
- * \return int -- 1 = success, 0 = failure
- * */
-int N_les_integrate_dirichlet_3d(N_les * les, N_geom_data * geom,
-				 N_array_3d * status, N_array_3d * start_val)
-{
-    int rows, cols, depths;
-    int count = 0;
-    int i, j, x, y, z, stat;
-    double *dvect1;
-    double *dvect2;
-
-    G_debug(2,
-	    "N_les_integrate_dirichlet_3d: integrating the dirichlet boundary condition");
-
-    rows = geom->rows;
-    cols = geom->cols;
-    depths = geom->depths;
-
-    /*we nned to additional vectors */
-    dvect1 = (double *)G_calloc(les->cols, sizeof(double));
-    dvect2 = (double *)G_calloc(les->cols, sizeof(double));
-
-    /*fill the first one with the x vector data of Dirichlet cells */
-    count = 0;
-    for (z = 0; z < depths; z++) {
-	for (y = 0; y < rows; y++) {
-	    for (x = 0; x < cols; x++) {
-		stat = (int)N_get_array_3d_d_value(status, x, y, z);
-		if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
-		    dvect1[count] =
-			N_get_array_3d_d_value(start_val, x, y, z);
-		    count++;
-		}
-		else if (stat == N_CELL_ACTIVE) {
-		    dvect1[count] = 0.0;
-		    count++;
-		}
-	    }
-	}
-    }
-
-#pragma omp parallel default(shared)
-    {
-	/*perform the matrix vector product and */
-	if (les->type == N_SPARSE_LES)
-	    G_math_Ax_sparse(les->Asp, dvect1, dvect2, les->rows);
-	else
-	    G_math_d_Ax(les->A, dvect1, dvect2, les->rows, les->cols);
-#pragma omp for schedule (static) private(i)
-	for (i = 0; i < les->cols; i++)
-	    les->b[i] = les->b[i] - dvect2[i];
-    }
-
-    /*now set the Dirichlet cell rows and cols to zero and the 
-     * diagonal entry to 1*/
-    count = 0;
-    for (z = 0; z < depths; z++) {
-	for (y = 0; y < rows; y++) {
-	    for (x = 0; x < cols; x++) {
-		stat = (int)N_get_array_3d_d_value(status, x, y, z);
-		if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
-		    if (les->type == N_SPARSE_LES) {
-			/*set the rows to zero */
-			for (i = 0; i < les->Asp[count]->cols; i++)
-			    les->Asp[count]->values[i] = 0.0;
-			/*set the cols to zero */
-			for (i = 0; i < les->rows; i++) {
-			    for (j = 0; j < les->Asp[i]->cols; j++) {
-				if (les->Asp[i]->index[j] == count)
-				    les->Asp[i]->values[j] = 0.0;
-			    }
-			}
-
-			/*entry on the diagonal */
-			les->Asp[count]->values[0] = 1.0;
-
-		    }
-		    else {
-			/*set the rows to zero */
-			for (i = 0; i < les->cols; i++)
-			    les->A[count][i] = 0.0;
-			/*set the cols to zero */
-			for (i = 0; i < les->rows; i++)
-			    les->A[i][count] = 0.0;
-
-			/*entry on the diagonal */
-			les->A[count][count] = 1.0;
-		    }
-		}
-		count++;
-	    }
-	}
-    }
-
-    return 0;
-
-}
-
-/* **************************************************************** */
-/* **** make an entry in the les (3d) ***************************** */
-/* **************************************************************** */
-int make_les_entry_3d(int i, int j, int k, int offset_i, int offset_j,
-		      int offset_k, int count, int pos, N_les * les,
-		      G_math_spvector * spvect, N_array_3d * cell_count,
-		      N_array_3d * status, N_array_3d * start_val,
-		      double entry, int cell_type)
-{
-    int K;
-    int di = offset_i;
-    int dj = offset_j;
-    int dk = offset_k;
-
-    K = (int)N_get_array_3d_d_value(cell_count, i + di, j + dj, k + dk) -
-	(int)N_get_array_3d_d_value(cell_count, i, j, k);
-
-    if (cell_type == N_CELL_ACTIVE) {
-	if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk) >
-	    N_CELL_ACTIVE &&
-	    (int)N_get_array_3d_d_value(status, i + di, j + dj,
-					k + dk) < N_MAX_CELL_STATE)
-	    les->b[count] -=
-		N_get_array_3d_d_value(start_val, i + di, j + dj,
-				       k + dk) * entry;
-	else if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk)
-		 == N_CELL_ACTIVE) {
-	    if ((count + K) >= 0 && (count + K) < les->cols) {
-		G_debug(5,
-			" make_les_entry_3d: (N_CELL_ACTIVE) create matrix entry at row[%i] col[%i] value %g\n",
-			count, count + K, entry);
-		pos++;
-		if (les->type == N_SPARSE_LES) {
-		    spvect->index[pos] = count + K;
-		    spvect->values[pos] = entry;
-		}
-		else {
-		    les->A[count][count + K] = entry;
-		}
-	    }
-	}
-    }
-    else if (cell_type == N_CELL_DIRICHLET) {
-	if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk)
-	    != N_CELL_INACTIVE) {
-	    if ((count + K) >= 0 && (count + K) < les->cols) {
-		G_debug(5,
-			" make_les_entry_3d: (N_CELL_DIRICHLET) create matrix entry at row[%i] col[%i] value %g\n",
-			count, count + K, entry);
-		pos++;
-		if (les->type == N_SPARSE_LES) {
-		    spvect->index[pos] = count + K;
-		    spvect->values[pos] = entry;
-		}
-		else {
-		    les->A[count][count + K] = entry;
-		}
-	    }
-	}
-    }
-
-    return pos;
-}

Deleted: grass/trunk/lib/gpde/N_parse_options.c
===================================================================
--- grass/trunk/lib/gpde/N_parse_options.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_parse_options.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,111 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      standard parser option for the numerical pde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <grass/glocale.h>
-#include <grass/N_pde.h>
-
-/*!
- * \brief Create standardised Option structure related to the gpde library.
- *
- * This function will create a standardised Option structure
- * defined by parameter opt. A list of valid parameters can be found in N_pde.h.
- * It allocates memory for the Option structure and returns a pointer to
- * this memory (of <i>type struct Option *</i>).<br>
- *
- * If an invalid parameter was specified an empty Option structure will
- * be returned (not NULL).
- *
- * This function is related to the gpde library, general standard options can be 
- * found in lib/gis/parser.c. These options are set with G_define_standard_option ();
- *
- * \param[in] opt Type of Option struct to create
- * \return Option * Pointer to an Option struct
- *
- * */
-struct Option *N_define_standard_option(int opt)
-{
-    struct Option *Opt;
-
-    Opt = G_define_option();
-
-    switch (opt) {
-	/*solver for symmetric, positive definite linear equation systems */
-    case N_OPT_SOLVER_SYMM:
-	Opt->key = "solver";
-	Opt->type = TYPE_STRING;
-	Opt->required = NO;
-	Opt->key_desc = "name";
-	Opt->answer = "cg";
-	Opt->options = "gauss,lu,cholesky,jacobi,sor,cg,bicgstab,pcg";
-        Opt->guisection = "Solver";
-	Opt->description =
-	    ("The type of solver which should solve the symmetric linear equation system");
-	break;
-	/*solver for unsymmetric linear equation systems */
-    case N_OPT_SOLVER_UNSYMM:
-	Opt->key = "solver";
-	Opt->type = TYPE_STRING;
-	Opt->required = NO;
-	Opt->key_desc = "name";
-	Opt->answer = "bicgstab";
-	Opt->options = "gauss,lu,jacobi,sor,bicgstab";
-        Opt->guisection = "Solver";
-	Opt->description =
-	    ("The type of solver which should solve the linear equation system");
-	break;
-    case N_OPT_MAX_ITERATIONS:
-	Opt->key = "maxit";
-	Opt->type = TYPE_INTEGER;
-	Opt->required = NO;
-	Opt->answer = "10000";
-        Opt->guisection = "Solver";
-	Opt->description =
-	    ("Maximum number of iteration used to solve the linear equation system");
-	break;
-    case N_OPT_ITERATION_ERROR:
-	Opt->key = "error";
-	Opt->type = TYPE_DOUBLE;
-	Opt->required = NO;
-	Opt->answer = "0.000001";
-        Opt->guisection = "Solver";
-	Opt->description =
-	    ("Error break criteria for iterative solver");
-	break;
-    case N_OPT_SOR_VALUE:
-	Opt->key = "relax";
-	Opt->type = TYPE_DOUBLE;
-	Opt->required = NO;
-	Opt->answer = "1";
-        Opt->guisection = "Solver";
-	Opt->description =
-	    ("The relaxation parameter used by the jacobi and sor solver for speedup or stabilizing");
-	break;
-    case N_OPT_CALC_TIME:
-	Opt->key = "dt";
-	Opt->type = TYPE_DOUBLE;
-	Opt->required = YES;
-	Opt->answer = "86400";
-        Opt->guisection = "Solver";
-	Opt->description = _("The calculation time in seconds");
-	break;
-    }
-
-    return Opt;
-}

Deleted: grass/trunk/lib/gpde/N_solute_transport.c
===================================================================
--- grass/trunk/lib/gpde/N_solute_transport.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_solute_transport.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,772 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      solute transport in porous media
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2007 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <math.h>
-#include <grass/N_solute_transport.h>
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*! \brief This is just a placeholder
- *
- * */
-N_data_star *N_callback_solute_transport_3d(void *solutedata,
-					    N_geom_data * geom, int col,
-					    int row, int depth)
-{
-    double Df_e = 0, Df_w = 0, Df_n = 0, Df_s = 0, Df_t = 0, Df_b = 0;
-    double dx, dy, dz, Az;
-    double diff_x, diff_y, diff_z;
-    double diff_xw, diff_yn;
-    double diff_xe, diff_ys;
-    double diff_zt, diff_zb;
-    double cin = 0, cg, cg_start;
-    double R, nf, cs, q;
-    double C, W, E, N, S, T, B, V;
-    double vw = 0, ve = 0, vn = 0, vs = 0, vt = 0, vb = 0;
-    double Ds_w = 0, Ds_e = 0, Ds_n = 0, Ds_s = 0, Ds_t = 0, Ds_b = 0;
-    double Dw = 0, De = 0, Dn = 0, Ds = 0, Dt = 0, Db = 0;
-    double rw = 0.5, re = 0.5, rn = 0.5, rs = 0.5, rt = 0.5, rb = 0.5;
-
-    N_solute_transport_data3d *data = NULL;
-    N_data_star *mat_pos;
-    N_gradient_3d grad;
-
-    /*cast the void pointer to the right data structure */
-    data = (N_solute_transport_data3d *) solutedata;
-
-    N_get_gradient_3d(data->grad, &grad, col, row, depth);
-
-    dx = geom->dx;
-    dy = geom->dy;
-    dz = geom->dz;
-    Az = N_get_geom_data_area_of_cell(geom, row);
-
-    /*read the data from the arrays */
-    cg_start = N_get_array_3d_d_value(data->c_start, col, row, depth);
-    cg = N_get_array_3d_d_value(data->c, col, row, depth);
-
-    /*get the surrounding diffusion tensor entries */
-    diff_x = N_get_array_3d_d_value(data->diff_x, col, row, depth);
-    diff_y = N_get_array_3d_d_value(data->diff_y, col, row, depth);
-    diff_z = N_get_array_3d_d_value(data->diff_z, col, row, depth);
-    diff_xw = N_get_array_3d_d_value(data->diff_x, col - 1, row, depth);
-    diff_xe = N_get_array_3d_d_value(data->diff_x, col + 1, row, depth);
-    diff_yn = N_get_array_3d_d_value(data->diff_y, col, row - 1, depth);
-    diff_ys = N_get_array_3d_d_value(data->diff_y, col, row + 1, depth);
-    diff_zt = N_get_array_3d_d_value(data->diff_z, col, row, depth + 1);
-    diff_zb = N_get_array_3d_d_value(data->diff_z, col, row, depth - 1);
-
-    /* calculate the diffusion on the cell borders using the harmonical mean */
-    Df_w = N_calc_harmonic_mean(diff_xw, diff_x);
-    Df_e = N_calc_harmonic_mean(diff_xe, diff_x);
-    Df_n = N_calc_harmonic_mean(diff_yn, diff_y);
-    Df_s = N_calc_harmonic_mean(diff_ys, diff_y);
-    Df_t = N_calc_harmonic_mean(diff_zt, diff_z);
-    Df_b = N_calc_harmonic_mean(diff_zb, diff_z);
-
-    /* calculate the dispersion */
-    /*todo */
-
-    /* calculate the velocity parts  with full upwinding scheme */
-    vw = grad.WC;
-    ve = grad.EC;
-    vn = grad.NC;
-    vs = grad.SC;
-    vt = grad.TC;
-    vb = grad.BC;
-
-    /* put the diffusion and dispersion together */
-    Dw = ((Df_w + Ds_w)) / dx;
-    De = ((Df_e + Ds_e)) / dx;
-    Dn = ((Df_n + Ds_n)) / dy;
-    Ds = ((Df_s + Ds_s)) / dy;
-    Dt = ((Df_t + Ds_t)) / dz;
-    Db = ((Df_b + Ds_b)) / dz;
-
-    rw = N_exp_upwinding(-1 * vw, dx, Dw);
-    re = N_exp_upwinding(ve, dx, De);
-    rs = N_exp_upwinding(-1 * vs, dy, Ds);
-    rn = N_exp_upwinding(vn, dy, Dn);
-    rb = N_exp_upwinding(-1 * vb, dz, Dn);
-    rt = N_exp_upwinding(vt, dz, Dn);
-
-    /*mass balance center cell to western cell */
-    W = -1 * (Dw) * dy * dz - vw * (1 - rw) * dy * dz;
-    /*mass balance center cell to eastern cell */
-    E = -1 * (De) * dy * dz + ve * (1 - re) * dy * dz;
-    /*mass balance center cell to southern cell */
-    S = -1 * (Ds) * dx * dz - vs * (1 - rs) * dx * dz;
-    /*mass balance center cell to northern cell */
-    N = -1 * (Dn) * dx * dz + vn * (1 - rn) * dx * dz;
-    /*mass balance center cell to bottom cell */
-    B = -1 * (Db) * Az - vb * (1 - rb) * Az;
-    /*mass balance center cell to top cell */
-    T = -1 * (Dt) * Az + vt * (1 - rt) * Az;
-
-    /* Retardation */
-    R = N_get_array_3d_d_value(data->R, col, row, depth);
-    /* Inner sources */
-    cs = N_get_array_3d_d_value(data->cs, col, row, depth);
-    /* effective porosity */
-    nf = N_get_array_3d_d_value(data->nf, col, row, depth);
-    /* groundwater sources and sinks */
-    q = N_get_array_3d_d_value(data->q, col, row, depth);
-    /* concentration of influent water */
-    cin = N_get_array_3d_d_value(data->cin, col, row, depth);
-
-    /*the diagonal entry of the matrix */
-    C = ((Dw - vw) * dy * dz +
-	 (De + ve) * dy * dz +
-	 (Ds - vs) * dx * dz +
-	 (Dn + vn) * dx * dz +
-	 (Db - vb) * Az + (Dt + vt) * Az + Az * dz * R / data->dt - q / nf);
-
-    /*the entry in the right side b of Ax = b */
-    V = (cs + cg_start * Az * dz * R / data->dt - q / nf * cin);
-
-    /*
-     * printf("nf %g\n", nf);
-     * printf("q %g\n", q);
-     * printf("cs %g\n", cs);
-     * printf("cin %g\n", cin);
-     * printf("cg %g\n", cg);
-     * printf("cg_start %g\n", cg_start);
-     * printf("Az %g\n", Az);
-     * printf("z %g\n", z);
-     * printf("R %g\n", R);
-     * printf("dt %g\n", data->dt);
-     */
-    G_debug(6, "N_callback_solute_transport_3d: called [%i][%i][%i]", row,
-	    col, depth);
-
-    /*create the 7 point star entries */
-    mat_pos = N_create_7star(C, W, E, N, S, T, B, V);
-
-    return mat_pos;
-}
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*!
- * \brief This callback function creates the mass balance of a 5 point star
- *
- * The mass balance is based on the common solute transport equation:
- *
- * \f[\frac{\partial c_g}{\partial t} R = \nabla \cdot ({\bf D} \nabla c_g - {\bf u} c_g) + \sigma + \frac{q}{n_f}(c_g - c_in) \f]
- *
- * This equation is discretizised with the finite volume method in two dimensions.
- *
- *
- * \param solutedata  * N_solute_transport_data2d - a void pointer to the data structure
- * \param geom N_geom_data *
- * \param col   int
- * \param row   int
- * \return N_data_star * - a five point data star
- *
- * */
-N_data_star *N_callback_solute_transport_2d(void *solutedata,
-					    N_geom_data * geom, int col,
-					    int row)
-{
-    double Df_e = 0, Df_w = 0, Df_n = 0, Df_s = 0;
-    double z_e = 0, z_w = 0, z_n = 0, z_s = 0;
-    double dx, dy, Az;
-    double diff_x, diff_y;
-    double disp_x, disp_y;
-    double z;
-    double diff_xw, diff_yn;
-    double disp_xw, disp_yn;
-    double z_xw, z_yn;
-    double diff_xe, diff_ys;
-    double disp_xe, disp_ys;
-    double z_xe, z_ys;
-    double cin = 0, cg, cg_start;
-    double R, nf, cs, q;
-    double C, W, E, N, S, V, NE, NW, SW, SE;
-    double vw = 0, ve = 0, vn = 0, vs = 0;
-    double Ds_w = 0, Ds_e = 0, Ds_n = 0, Ds_s = 0;
-    double Dw = 0, De = 0, Dn = 0, Ds = 0;
-    double rw = 0.5, re = 0.5, rn = 0.5, rs = 0.5;
-
-    N_solute_transport_data2d *data = NULL;
-    N_data_star *mat_pos;
-    N_gradient_2d grad;
-
-    /*cast the void pointer to the right data structure */
-    data = (N_solute_transport_data2d *) solutedata;
-
-    N_get_gradient_2d(data->grad, &grad, col, row);
-
-    dx = geom->dx;
-    dy = geom->dy;
-    Az = N_get_geom_data_area_of_cell(geom, row);
-
-    /*read the data from the arrays */
-    cg_start = N_get_array_2d_d_value(data->c_start, col, row);
-    cg = N_get_array_2d_d_value(data->c, col, row);
-
-    /* calculate the cell height */
-    z = N_get_array_2d_d_value(data->top, col,
-			       row) -
-	N_get_array_2d_d_value(data->bottom, col, row);
-    z_xw =
-	N_get_array_2d_d_value(data->top, col - 1,
-			       row) -
-	N_get_array_2d_d_value(data->bottom, col - 1, row);
-    z_xe =
-	N_get_array_2d_d_value(data->top, col + 1,
-			       row) -
-	N_get_array_2d_d_value(data->bottom, col + 1, row);
-    z_yn =
-	N_get_array_2d_d_value(data->top, col,
-			       row - 1) -
-	N_get_array_2d_d_value(data->bottom, col, row - 1);
-    z_ys =
-	N_get_array_2d_d_value(data->top, col,
-			       row + 1) -
-	N_get_array_2d_d_value(data->bottom, col, row + 1);
-
-    /*geometrical mean of cell height */
-    z_w = N_calc_geom_mean(z_xw, z);
-    z_e = N_calc_geom_mean(z_xe, z);
-    z_n = N_calc_geom_mean(z_yn, z);
-    z_s = N_calc_geom_mean(z_ys, z);
-
-    /*get the surrounding diffusion tensor entries */
-    diff_x = N_get_array_2d_d_value(data->diff_x, col, row);
-    diff_y = N_get_array_2d_d_value(data->diff_y, col, row);
-    diff_xw = N_get_array_2d_d_value(data->diff_x, col - 1, row);
-    diff_xe = N_get_array_2d_d_value(data->diff_x, col + 1, row);
-    diff_yn = N_get_array_2d_d_value(data->diff_y, col, row - 1);
-    diff_ys = N_get_array_2d_d_value(data->diff_y, col, row + 1);
-
-    /* calculate the diffusion at the cell borders using the harmonical mean */
-    Df_w = N_calc_harmonic_mean(diff_xw, diff_x);
-    Df_e = N_calc_harmonic_mean(diff_xe, diff_x);
-    Df_n = N_calc_harmonic_mean(diff_yn, diff_y);
-    Df_s = N_calc_harmonic_mean(diff_ys, diff_y);
-
-    /* calculate the dispersion */
-    /*get the surrounding dispersion tensor entries */
-    disp_x = N_get_array_2d_d_value(data->disp_xx, col, row);
-    disp_y = N_get_array_2d_d_value(data->disp_yy, col, row);
-    if (N_get_array_2d_d_value(data->status, col - 1, row) ==
-	N_CELL_TRANSMISSION) {
-	disp_xw = disp_x;
-    }
-    else {
-	disp_xw = N_get_array_2d_d_value(data->disp_xx, col - 1, row);
-    }
-    if (N_get_array_2d_d_value(data->status, col + 1, row) ==
-	N_CELL_TRANSMISSION) {
-	disp_xe = disp_x;
-    }
-    else {
-	disp_xe = N_get_array_2d_d_value(data->disp_xx, col + 1, row);
-    }
-    if (N_get_array_2d_d_value(data->status, col, row - 1) ==
-	N_CELL_TRANSMISSION) {
-	disp_yn = disp_y;
-    }
-    else {
-	disp_yn = N_get_array_2d_d_value(data->disp_yy, col, row - 1);
-    }
-    if (N_get_array_2d_d_value(data->status, col, row + 1) ==
-	N_CELL_TRANSMISSION) {
-	disp_ys = disp_y;
-    }
-    else {
-	disp_ys = N_get_array_2d_d_value(data->disp_yy, col, row + 1);
-    }
-
-    /* calculate the dispersion at the cell borders using the harmonical mean */
-    Ds_w = N_calc_harmonic_mean(disp_xw, disp_x);
-    Ds_e = N_calc_harmonic_mean(disp_xe, disp_x);
-    Ds_n = N_calc_harmonic_mean(disp_yn, disp_y);
-    Ds_s = N_calc_harmonic_mean(disp_ys, disp_y);
-
-    /* put the diffusion and dispersion together */
-    Dw = ((Df_w + Ds_w)) / dx;
-    De = ((Df_e + Ds_e)) / dx;
-    Ds = ((Df_s + Ds_s)) / dy;
-    Dn = ((Df_n + Ds_n)) / dy;
-
-    vw = -1.0 * grad.WC;
-    ve = grad.EC;
-    vs = -1.0 * grad.SC;
-    vn = grad.NC;
-
-    if (data->stab == N_UPWIND_FULL) {
-	rw = N_full_upwinding(vw, dx, Dw);
-	re = N_full_upwinding(ve, dx, De);
-	rs = N_full_upwinding(vs, dy, Ds);
-	rn = N_full_upwinding(vn, dy, Dn);
-    }
-    else if (data->stab == N_UPWIND_EXP) {
-	rw = N_exp_upwinding(vw, dx, Dw);
-	re = N_exp_upwinding(ve, dx, De);
-	rs = N_exp_upwinding(vs, dy, Ds);
-	rn = N_exp_upwinding(vn, dy, Dn);
-    }
-
-    /*mass balance center cell to western cell */
-    W = -1 * (Dw) * dy * z_w + vw * (1 - rw) * dy * z_w;
-    /*mass balance center cell to eastern cell */
-    E = -1 * (De) * dy * z_e + ve * (1 - re) * dy * z_e;
-    /*mass balance center cell to southern cell */
-    S = -1 * (Ds) * dx * z_s + vs * (1 - rs) * dx * z_s;
-    /*mass balance center cell to northern cell */
-    N = -1 * (Dn) * dx * z_n + vn * (1 - rn) * dx * z_n;
-
-    NW = 0.0;
-    SW = 0.0;
-    NE = 0.0;
-    SE = 0.0;
-
-    /* Retardation */
-    R = N_get_array_2d_d_value(data->R, col, row);
-    /* Inner sources */
-    cs = N_get_array_2d_d_value(data->cs, col, row);
-    /* effective porosity */
-    nf = N_get_array_2d_d_value(data->nf, col, row);
-    /* groundwater sources and sinks */
-    q = N_get_array_2d_d_value(data->q, col, row);
-    /* concentration of influent water */
-    cin = N_get_array_2d_d_value(data->cin, col, row);
-
-    /*the diagonal entry of the matrix */
-    C = (Dw + vw * rw) * dy * z_w +
-	(De + ve * re) * dy * z_e +
-	(Ds + vs * rs) * dx * z_s +
-	(Dn + vn * rn) * dx * z_n + Az * z * R / data->dt - q / nf;
-
-    /*the entry in the right side b of Ax = b */
-    V = (cs + cg_start * Az * z * R / data->dt + q / nf * cin);
-
-    /*
-       fprintf(stderr, "nf %g\n", nf);
-       fprintf(stderr, "q %g\n", q);
-       fprintf(stderr, "cs %g\n", cs);
-       fprintf(stderr, "cin %g\n", cin);
-       fprintf(stderr, "cg %g\n", cg);
-       fprintf(stderr, "cg_start %g\n", cg_start);
-       fprintf(stderr, "Az %g\n", Az);
-       fprintf(stderr, "z %g\n", z);
-       fprintf(stderr, "R %g\n", R);
-       fprintf(stderr, "dt %g\n", data->dt);
-     */
-
-    G_debug(6, "N_callback_solute_transport_2d: called [%i][%i]", row, col);
-
-    /*create the 9 point star entries */
-    mat_pos = N_create_9star(C, W, E, N, S, NW, SW, NE, SE, V);
-
-    return mat_pos;
-}
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*!
- * \brief Alllocate memory for the solute transport data structure in three dimensions
- *
- * The solute transport data structure will be allocated including
- * all appendant 3d arrays. The offset for the 3d arrays is one
- * to establish homogeneous Neumann boundary conditions at the calculation area border.
- * This data structure is used to create a linear equation system based on the computation of
- * solute transport in porous media with the finite volume method.
- *
- * \param cols   int
- * \param rows   int
- * \param depths int
- * \return N_solute_transport_data3d *
- * */
-
-N_solute_transport_data3d *N_alloc_solute_transport_data3d(int cols, int rows,
-							   int depths)
-{
-    N_solute_transport_data3d *data = NULL;
-
-    data =
-	(N_solute_transport_data3d *) G_calloc(1,
-					       sizeof
-					       (N_solute_transport_data3d));
-
-    data->c = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->c_start = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->status = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->diff_x = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->diff_y = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->diff_z = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->q = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->cs = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->R = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->nf = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->cin = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-
-    /*Allocate the dispersivity tensor */
-    data->disp_xx = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->disp_yy = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->disp_zz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->disp_xy = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->disp_xz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-    data->disp_yz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
-
-
-    data->grad = N_alloc_gradient_field_3d(cols, rows, depths);
-    data->stab = N_UPWIND_EXP;
-
-    return data;
-}
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*!
- * \brief Alllocate memory for the solute transport data structure in two dimensions
- *
- * The solute transport data structure will be allocated including
- * all appendant 2d arrays. The offset for the 2d arrays is one
- * to establish homogeneous Neumann boundary conditions at the calculation area border.
- * This data structure is used to create a linear equation system based on the computation of
- * solute transport in porous media with the finite volume method.
- *
- * \param cols   int
- * \param rows   int
- * \return N_solute_transport_data2d *
- * */
-
-
-N_solute_transport_data2d *N_alloc_solute_transport_data2d(int cols, int rows)
-{
-    N_solute_transport_data2d *data = NULL;
-
-    data =
-	(N_solute_transport_data2d *) G_calloc(1,
-					       sizeof
-					       (N_solute_transport_data2d));
-
-    data->c = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->c_start = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->status = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->diff_x = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->diff_y = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->q = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->cs = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->R = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->nf = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->cin = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->top = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->bottom = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-
-    /*Allocate the dispersivity tensor */
-    data->disp_xx = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->disp_yy = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-    data->disp_xy = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
-
-    data->grad = N_alloc_gradient_field_2d(cols, rows);
-    data->stab = N_UPWIND_EXP;
-
-    return data;
-}
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*!
- * \brief Release the memory of the solute transport data structure in three dimensions
- *
- * \param data N_solute_transport_data2d *
- * \return void *
- * */
-void N_free_solute_transport_data3d(N_solute_transport_data3d * data)
-{
-    N_free_array_3d(data->c);
-    N_free_array_3d(data->c_start);
-    N_free_array_3d(data->status);
-    N_free_array_3d(data->diff_x);
-    N_free_array_3d(data->diff_y);
-    N_free_array_3d(data->diff_z);
-    N_free_array_3d(data->q);
-    N_free_array_3d(data->cs);
-    N_free_array_3d(data->R);
-    N_free_array_3d(data->nf);
-    N_free_array_3d(data->cin);
-
-    N_free_array_3d(data->disp_xx);
-    N_free_array_3d(data->disp_yy);
-    N_free_array_3d(data->disp_zz);
-    N_free_array_3d(data->disp_xy);
-    N_free_array_3d(data->disp_xz);
-    N_free_array_3d(data->disp_yz);
-
-    G_free(data);
-
-    data = NULL;
-
-    return;
-}
-
-/* ************************************************************************* *
- * ************************************************************************* *
- * ************************************************************************* */
-/*!
- * \brief Release the memory of the solute transport data structure in two dimensions
- *
- * \param data N_solute_transport_data2d *
- * \return void *
- * */
-void N_free_solute_transport_data2d(N_solute_transport_data2d * data)
-{
-    N_free_array_2d(data->c);
-    N_free_array_2d(data->c_start);
-    N_free_array_2d(data->status);
-    N_free_array_2d(data->diff_x);
-    N_free_array_2d(data->diff_y);
-    N_free_array_2d(data->q);
-    N_free_array_2d(data->cs);
-    N_free_array_2d(data->R);
-    N_free_array_2d(data->nf);
-    N_free_array_2d(data->cin);
-    N_free_array_2d(data->top);
-    N_free_array_2d(data->bottom);
-
-    N_free_array_2d(data->disp_xx);
-    N_free_array_2d(data->disp_yy);
-    N_free_array_2d(data->disp_xy);
-
-    G_free(data);
-
-    data = NULL;
-
-    return;
-}
-
-/*!
- * \brief Compute the transmission boundary condition in 2d
- *
- * This function calculates the transmission boundary condition
- * for each cell with status N_CELL_TRANSMISSION. The surrounding
- * gradient field is used to verfiy the flow direction. If a flow
- * goes into a cell, the concentration (data->c) from the neighbour cell is
- * added to the transmission cell. If the flow from several neighbour 
- * cells goes into the cell, the concentration mean is calculated.
- * 
- * The new concentrations are written into the data->c_start array,
- * so they can be handled by the matrix assembling function.
- *
- * \param data N_solute_transport_data2d *
- * \return void *
- * */
-void N_calc_solute_transport_transmission_2d(N_solute_transport_data2d * data)
-{
-    int i, j, count = 1;
-    int cols, rows;
-    double c;
-    N_gradient_2d grad;
-
-    cols = data->grad->cols;
-    rows = data->grad->rows;
-
-    G_debug(2,
-	    "N_calc_solute_transport_transmission_2d: calculating transmission boundary");
-
-    for (j = 0; j < rows; j++) {
-	for (i = 0; i < cols; i++) {
-	    if (N_get_array_2d_d_value(data->status, i, j) ==
-		N_CELL_TRANSMISSION) {
-		count = 0;
-		/*get the gradient neighbours */
-		N_get_gradient_2d(data->grad, &grad, i, j);
-		c = 0;
-		/*
-		   c = N_get_array_2d_d_value(data->c_start, i, j);
-		   if(c > 0)
-		   count++;
-		 */
-
-		if (grad.WC > 0 &&
-		    !N_is_array_2d_value_null(data->c, i - 1, j)) {
-		    c += N_get_array_2d_d_value(data->c, i - 1, j);
-		    count++;
-		}
-		if (grad.EC < 0 &&
-		    !N_is_array_2d_value_null(data->c, i + 1, j)) {
-		    c += N_get_array_2d_d_value(data->c, i + 1, j);
-		    count++;
-		}
-		if (grad.NC < 0 &&
-		    !N_is_array_2d_value_null(data->c, i, j - 1)) {
-		    c += N_get_array_2d_d_value(data->c, i, j - 1);
-		    count++;
-		}
-		if (grad.SC > 0 &&
-		    !N_is_array_2d_value_null(data->c, i, j + 1)) {
-		    c += N_get_array_2d_d_value(data->c, i, j + 1);
-		    count++;
-		}
-		if (count != 0)
-		    c = c / (double)count;
-		/*make sure it is not NAN */
-		if (c > 0 || c == 0 || c < 0)
-		    N_put_array_2d_d_value(data->c_start, i, j, c);
-	    }
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Compute the dispersivity tensor based on the solute transport data in 2d
- *
- * The dispersivity tensor is stored in the data structure.
- * To compute the dispersivity tensor, the dispersivity lentghs and the gradient field
- * must be present.
- *
- * This is just a simple tensor computation which should be extended.
- *
- * \todo Change the tensor calculation to a mor realistic algorithm 
- *
- * \param data N_solute_transport_data2d *
- * \return void *
- * */
-void N_calc_solute_transport_disptensor_2d(N_solute_transport_data2d * data)
-{
-    int i, j;
-    int cols, rows;
-    double vx, vy, vv;
-    double disp_xx, disp_yy, disp_xy;
-    N_gradient_2d grad;
-
-    cols = data->grad->cols;
-    rows = data->grad->rows;
-
-    G_debug(2,
-	    "N_calc_solute_transport_disptensor_2d: calculating the dispersivity tensor");
-
-    for (j = 0; j < rows; j++) {
-	for (i = 0; i < cols; i++) {
-
-	    disp_xx = 0;
-	    disp_yy = 0;
-	    disp_xy = 0;
-
-	    /*get the gradient neighbours */
-	    N_get_gradient_2d(data->grad, &grad, i, j);
-	    vx = (grad.WC + grad.EC) / 2;
-	    vy = (grad.NC + grad.SC) / 2;
-	    vv = sqrt(vx * vx + vy * vy);
-
-	    if (vv != 0) {
-		disp_xx = data->al * vx * vx / vv + data->at * vy * vy / vv;
-		disp_yy = data->at * vx * vx / vv + data->al * vy * vy / vv;
-		disp_xy = (data->al - data->at) * vx * vy / vv;
-	    }
-
-	    G_debug(5,
-		    "N_calc_solute_transport_disptensor_2d: [%i][%i] disp_xx %g disp_yy %g disp_xy %g",
-		    i, j, disp_xx, disp_yy, disp_xy);
-	    N_put_array_2d_d_value(data->disp_xx, i, j, disp_xx);
-	    N_put_array_2d_d_value(data->disp_yy, i, j, disp_yy);
-	    N_put_array_2d_d_value(data->disp_xy, i, j, disp_xy);
-	}
-    }
-
-    return;
-}
-
-/*!
- * \brief Compute the dispersivity tensor based on the solute transport data in 3d
- *
- * The dispersivity tensor is stored in the data structure.
- * To compute the dispersivity tensor, the dispersivity lentghs and the gradient field
- * must be present.
- * 
- * This is just a simple tensor computation which should be extended.
- *
- * \todo Change the tensor calculation to a mor realistic algorithm 
- *
- * \param data N_solute_transport_data3d *
- * \return void *
- * */
-void N_calc_solute_transport_disptensor_3d(N_solute_transport_data3d * data)
-{
-    int i, j, k;
-    int cols, rows, depths;
-    double vx, vy, vz, vv;
-    double disp_xx, disp_yy, disp_zz, disp_xy, disp_xz, disp_yz;
-    N_gradient_3d grad;
-
-    cols = data->grad->cols;
-    rows = data->grad->rows;
-    depths = data->grad->depths;
-
-    G_debug(2,
-	    "N_calc_solute_transport_disptensor_3d: calculating the dispersivity tensor");
-
-    for (k = 0; k < depths; k++) {
-	for (j = 0; j < rows; j++) {
-	    for (i = 0; i < cols; i++) {
-		disp_xx = 0;
-		disp_yy = 0;
-		disp_zz = 0;
-		disp_xy = 0;
-		disp_xz = 0;
-		disp_yz = 0;
-
-		/*get the gradient neighbours */
-		N_get_gradient_3d(data->grad, &grad, i, j, k);
-		vx = (grad.WC + grad.EC) / 2;
-		vy = (grad.NC + grad.SC) / 2;
-		vz = (grad.BC + grad.TC) / 2;
-		vv = sqrt(vx * vx + vy * vy + vz * vz);
-
-		if (vv != 0) {
-		    disp_xx =
-			data->al * vx * vx / vv + data->at * vy * vy / vv +
-			data->at * vz * vz / vv;
-		    disp_yy =
-			data->at * vx * vx / vv + data->al * vy * vy / vv +
-			data->at * vz * vz / vv;
-		    disp_zz =
-			data->at * vx * vx / vv + data->at * vy * vy / vv +
-			data->al * vz * vz / vv;
-		    disp_xy = (data->al - data->at) * vx * vy / vv;
-		    disp_xz = (data->al - data->at) * vx * vz / vv;
-		    disp_yz = (data->al - data->at) * vy * vz / vv;
-		}
-
-		G_debug(5,
-			"N_calc_solute_transport_disptensor_3d: [%i][%i][%i] disp_xx %g disp_yy %g disp_zz %g  disp_xy %g disp_xz %g disp_yz %g ",
-			i, j, k, disp_xx, disp_yy, disp_zz, disp_xy, disp_xz,
-			disp_yz);
-		N_put_array_3d_d_value(data->disp_xx, i, j, k, disp_xx);
-		N_put_array_3d_d_value(data->disp_yy, i, j, k, disp_yy);
-		N_put_array_3d_d_value(data->disp_zz, i, j, k, disp_zz);
-		N_put_array_3d_d_value(data->disp_xy, i, j, k, disp_xy);
-		N_put_array_3d_d_value(data->disp_xz, i, j, k, disp_xz);
-		N_put_array_3d_d_value(data->disp_yz, i, j, k, disp_yz);
-	    }
-	}
-    }
-
-    return;
-}

Deleted: grass/trunk/lib/gpde/N_tools.c
===================================================================
--- grass/trunk/lib/gpde/N_tools.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_tools.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,200 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:     	Array managment functions 
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <math.h>
-#include <grass/N_pde.h>
-#include <grass/glocale.h>
-
-
-/*!
- * \brief Calculate the arithmetic mean of values a and b
- *
- * mean = (a+b)/2
- *
- * \param a double
- * \param b double
- * \return val double
- * */
-double N_calc_arith_mean(double a, double b)
-{
-    double val = 0;
-
-    val = (a + b) / 2.0;
-
-    return val;
-}
-
-/*!
- * \brief Calculate the arithmetic mean of the values in vector a
- * of size n
- *
- * n = [0 ... size[
- * mean =  (a[0] + a[1] + ... + a[n])/size
- *
- * \param a double * -- the value vector 
- * \param size int -- the size of the vector a
- * \return val double
- * */
-double N_calc_arith_mean_n(double *a, int size)
-{
-    double val = 0.0;
-    int i;
-
-    for (i = 0; i < size; i++)
-	val += a[i];
-
-    val = (val / (double)size);
-
-    return val;
-}
-
-
-/*!
- * \brief Calculate the geometrical mean of values a and b
- *
- * mean = sqrt(a*b)
- *
- * \param a double
- * \param b double
- * \return val double
- * */
-double N_calc_geom_mean(double a, double b)
-{
-    double val = 0;
-
-    val = sqrt(a * b);
-
-    return val;
-}
-
-/*!
- * \brief Calculate the geometrical mean of the values in vector a
- * of size n
- *
- * n = [0 ... size[
- * mean =  pow((a[0] * a[1] * ... * a[n]), 1.0/size)
- *
- * \param a double * -- the value vector 
- * \param size int -- the size of the vector a
- * \return val double
- * */
-double N_calc_geom_mean_n(double *a, int size)
-{
-    double val = 1;
-    int i;
-
-    for (i = 0; i < size; i++)
-	val *= a[i];
-
-    val = (double)pow((long double)val, (long double)1.0 / (long double)size);
-
-    return val;
-}
-
-
-/*!
- * \brief Calculate the harmonical mean of values a and b
- *
- * mean = 2*(a*b)/(a + b)
- *
- * \param a double
- * \param b double
- * \return val double -- if (a + b) == 0, a 0 is returned
- * */
-double N_calc_harmonic_mean(double a, double b)
-{
-    double val = 0.0;
-
-    if ((a + b) != 0)
-	val = 2.0 * (a * b) / (a + b);
-
-    return val;
-}
-
-/*!
- * \brief Calculate the harmonical mean of the values in vector a
- * of size n
- *
- * n = [0 ... size[
- * mean = 1/(1/size *(1/a[0] + 1/a[1] + ... + 1/a[n]))
- * 
- * \param a double * -- the value vector 
- * \param size int -- the size of the vector a
- * \return val double -- if one division with 0 is detected, 0 will be returned
- * */
-double N_calc_harmonic_mean_n(double *a, int size)
-{
-    double val = 0;
-    int i;
-
-    for (i = 0; i < size; i++)
-	if (a[i] != 0.0)
-	    val += 1.0 / a[i];
-	else
-	    return 0.0;
-
-    if (val == 0.0)
-	return 0.0;
-    else
-	val = 1.0 / (1.0 / (double)size * val);
-
-    return val;
-}
-
-
-/*!
- * \brief Calculate the quadratic mean of values a and b
- *
- * mean = sqrt((a*a + b*b)/2)
- *
- * \param a double
- * \param b double
- * \return val double 
- * */
-double N_calc_quad_mean(double a, double b)
-{
-    double val = 0.0;
-
-    val = sqrt((a * a + b * b) / 2.0);
-
-    return val;
-}
-
-/*!
- * \brief Calculate the quadratic mean of the values in vector a
- * of size n
- *
- * n = [0 ... size[
- * mean = sqrt((a[0]*a[0] + a[1]*a[1] + ... + a[n]*a[n])/size)
- * 
- * \param a double * -- the value vector 
- * \param size int -- the size of the vector a
- * \return val double 
- * */
-double N_calc_quad_mean_n(double *a, int size)
-{
-    double val = 0;
-    int i;
-
-    for (i = 0; i < size; i++)
-	val += a[i] * a[i];
-
-    val = sqrt(val / (double)size);
-
-    return val;
-}

Deleted: grass/trunk/lib/gpde/N_upwind.c
===================================================================
--- grass/trunk/lib/gpde/N_upwind.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/gpde/N_upwind.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,77 +0,0 @@
-
-/*****************************************************************************
-*
-* MODULE:       Grass PDE Numerical Library
-* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
-* 		soerengebbert <at> gmx <dot> de
-*               
-* PURPOSE:      upwinding stabilization algorithms
-* 		part of the gpde library
-*
-* COPYRIGHT:    (C) 2000 by the GRASS Development Team
-*
-*               This program is free software under the GNU General Public
-*               License (>=v2). Read the file COPYING that comes with GRASS
-*               for details.
-*
-*****************************************************************************/
-
-#include <math.h>
-#include <grass/N_pde.h>
-
-
-/*! \brief full upwinding stabilization algorithm
- *
- * The arguments are values to compute the local peclet number
- *
- * \param sprod double -- the scalar produkt between the velocity vector and the normal vector between two points
- * \param distance double -- distance between two points
- * \param D double -- diffusion/dispersion tensor part between two points
- *
- * \return the weighting factor
- * */
-double N_full_upwinding(double sprod, double distance, double D)
-{
-    double z;
-
-    if (D == 0)
-	return 0.5;
-
-    /*compute the local peclet number */
-    z = sprod * distance / D;
-
-    if (z > 0)
-	return 1;
-    if (z == 0)
-	return 0.5;
-    if (z < 0)
-	return 0;
-
-    return 0;
-}
-
-/*! \brief exponential upwinding stabilization algorithm
- *
- * The arguments are values to compute the local peclet number
- *
- * \param sprod double -- the scalar produkt between the velocity vector and the normal vector between two points
- * \param distance double -- distance between two points
- * \param D double -- diffusion/dispersion tensor part between two points
- *
- * \return the weighting factor
- * */
-double N_exp_upwinding(double sprod, double distance, double D)
-{
-    double z;
-
-    if (D == 0)
-	return 0.5;
-
-    /*compute the local peclet number */
-    z = sprod * distance / D;
-
-    if (z != 0)
-	return (1 - (1 / z) * (1 - (z / (exp(z) - 1))));
-
-    return 0.5;
-}

Copied: grass/trunk/lib/gpde/n_arrays.c (from rev 62429, grass/trunk/lib/gpde/N_arrays.c)
===================================================================
--- grass/trunk/lib/gpde/n_arrays.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_arrays.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,1244 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*
+* PURPOSE:     	Array managment functions
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <math.h>
+
+#include <grass/N_pde.h>
+#include <grass/raster.h>
+#include <grass/glocale.h>
+
+
+/* ******************** 2D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Allocate memory for a N_array_2d data structure.
+ *
+ * This function allocates memory for an array of type N_array_2d
+ * and returns the pointer to the new allocated memory.
+ * <br><br>
+ * The data type of this array is set by "type" and must be
+ * CELL_TYPE, FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
+ * The offset sets the number of boundary cols and rows.
+ * This option is useful to generate homogeneous Neumann boundary conditions around
+ * an array or to establish overlapping boundaries. The array is initialized with 0 by default.
+ * <br><br>
+ * If the offset is greater then 0, negative indices are possible.
+ * <br><br>
+ *
+ * The data structure of a array with 3 rows and cols and an offset of 1
+ * will looks like this:
+ * <br><br>
+ *
+ \verbatim
+ 0 0 0 0 0
+ 0 0 1 2 0
+ 0 3 4 5 0
+ 0 6 7 8 0
+ 0 0 0 0 0
+ \endverbatim
+ *
+ * 0 is the boundary.
+ * <br><br>
+ * Internal a one dimensional array is allocated to save memory and to speed up the memory access.
+ * To access the one dimensional array with a two dimensional index use the provided
+ * get and put functions. The internal representation of the above data will look like this:
+ *
+ \verbatim
+ 0 0 0 0 0 0 0 1 2 0 0 3 4 5 0 0 6 7 8 0 0 0 0 0 0
+ \endverbatim
+ *
+ * \param cols int
+ * \param rows int
+ * \param offset int
+ * \param type int
+ * \return N_array_2d *
+ *
+ * */
+N_array_2d *N_alloc_array_2d(int cols, int rows, int offset, int type)
+{
+    N_array_2d *data = NULL;
+
+    if (rows < 1 || cols < 1)
+	G_fatal_error("N_alloc_array_2d: cols and rows should be > 0");
+
+    if (type != CELL_TYPE && type != FCELL_TYPE && type != DCELL_TYPE)
+	G_fatal_error
+	    ("N_alloc_array_2d: Wrong data type, should be CELL_TYPE, FCELL_TYPE or DCELL_TYPE");
+
+    data = (N_array_2d *) G_calloc(1, sizeof(N_array_2d));
+
+    data->cols = cols;
+    data->rows = rows;
+    data->type = type;
+    data->offset = offset;
+    data->rows_intern = rows + 2 * offset;	/*offset position at booth sides */
+    data->cols_intern = cols + 2 * offset;	/*offset position at booth sides */
+    data->cell_array = NULL;
+    data->fcell_array = NULL;
+    data->dcell_array = NULL;
+
+    if (data->type == CELL_TYPE) {
+	data->cell_array =
+	    (CELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
+			      sizeof(CELL));
+	G_debug(3,
+		"N_alloc_array_2d: CELL array allocated rows_intern %i cols_intern %i offset %i",
+		data->rows_intern, data->cols_intern, data->offset = offset);
+    }
+    else if (data->type == FCELL_TYPE) {
+	data->fcell_array =
+	    (FCELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
+			       sizeof(FCELL));
+	G_debug(3,
+		"N_alloc_array_2d: FCELL array allocated rows_intern %i cols_intern %i offset %i",
+		data->rows_intern, data->cols_intern, data->offset = offset);
+
+    }
+    else if (data->type == DCELL_TYPE) {
+	data->dcell_array =
+	    (DCELL *) G_calloc((size_t) data->rows_intern * data->cols_intern,
+			       sizeof(DCELL));
+	G_debug(3,
+		"N_alloc_array_2d: DCELL array allocated rows_intern %i cols_intern %i offset %i",
+		data->rows_intern, data->cols_intern, data->offset = offset);
+    }
+
+    return data;
+}
+
+/*!
+ * \brief Release the memory of a N_array_2d structure
+ *
+ * \param data N_array_2d *
+ * \return void
+ * */
+void N_free_array_2d(N_array_2d * data)
+{
+
+    if (data != NULL) {
+	G_debug(3, "N_free_array_2d: free N_array_2d");
+
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    G_free(data->cell_array);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_free(data->fcell_array);
+
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_free(data->dcell_array);
+	}
+
+	G_free(data);
+	data = NULL;
+
+    }
+
+    return;
+}
+
+
+/*!
+ * \brief Return the data type of the N_array_2d struct
+ *
+ * The data type can be CELL_TYPE, FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
+ *
+ * \param array N_array_2d *
+ * \return type int
+ * */
+int N_get_array_2d_type(N_array_2d * array)
+{
+    return array->type;
+}
+
+/*!
+ * \brief Write the value of the N_array_2d struct at position col, row to value
+ *
+ * The value must be of the same type as the array. Otherwise you will risk data losses.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \param value void * - this variable contains the array value at col, row position
+ * \return void
+ * */
+
+void N_get_array_2d_value(N_array_2d * data, int col, int row, void *value)
+{
+
+    if (data->offset == 0) {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    *((CELL *) value) =
+		data->cell_array[row * data->cols_intern + col];
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    *((FCELL *) value) =
+		data->fcell_array[row * data->cols_intern + col];
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    *((DCELL *) value) =
+		data->dcell_array[row * data->cols_intern + col];
+	}
+    }
+    else {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    *((CELL *) value) =
+		data->cell_array[(row + data->offset) * data->cols_intern +
+				 col + data->offset];
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    *((FCELL *) value) =
+		data->fcell_array[(row + data->offset) * data->cols_intern +
+				  col + data->offset];
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    *((DCELL *) value) =
+		data->dcell_array[(row + data->offset) * data->cols_intern +
+				  col + data->offset];
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Returns 1 if the value of N_array_2d struct at postion col, row
+ * is of type null, otherwise 0
+ *
+ * This function checks automatically the type of the array and checks for the
+ * data type null value.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \return int - 1 = is null, 0 otherwise
+ * */
+int N_is_array_2d_value_null(N_array_2d * data, int col, int row)
+{
+
+    if (data->offset == 0) {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type CELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     cell_array[row * data->cols_intern +
+						col]), CELL_TYPE);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type FCELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     fcell_array[row * data->cols_intern +
+						 col]), FCELL_TYPE);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type DCELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     dcell_array[row * data->cols_intern +
+						 col]), DCELL_TYPE);
+	}
+    }
+    else {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type CELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     cell_array[(row +
+						 data->offset) *
+						data->cols_intern + col +
+						data->offset]), CELL_TYPE);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type FCELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     fcell_array[(row +
+						  data->offset) *
+						 data->cols_intern + col +
+						 data->offset]), FCELL_TYPE);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_2d_value_null: null value is of type DCELL at pos [%i][%i]",
+		    col, row);
+	    return Rast_is_null_value((void *)
+				   &(data->
+				     dcell_array[(row +
+						  data->offset) *
+						 data->cols_intern + col +
+						 data->offset]), DCELL_TYPE);
+	}
+    }
+
+    return 0;
+}
+
+
+/*!
+ * \brief Returns the value of type CELL at position col, row
+ *
+ * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the CELL type.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \return CELL
+ *
+ * */
+CELL N_get_array_2d_c_value(N_array_2d * data, int col, int row)
+{
+    CELL value = 0;
+    FCELL fvalue = 0.0;
+    DCELL dvalue = 0.0;
+
+    switch (data->type) {
+    case CELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&value);
+	return (CELL) value;
+    case FCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&fvalue);
+	return (CELL) fvalue;
+    case DCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&dvalue);
+	return (CELL) dvalue;
+    }
+
+    return value;
+}
+
+/*!
+ * \brief Returns the value of type FCELL at position col, row
+ *
+ * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the FCELL type.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \return FCELL
+
+ * */
+FCELL N_get_array_2d_f_value(N_array_2d * data, int col, int row)
+{
+    CELL value = 0;
+    FCELL fvalue = 0.0;
+    DCELL dvalue = 0.0;
+
+    switch (data->type) {
+    case CELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&value);
+	return (FCELL) value;
+    case FCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&fvalue);
+	return (FCELL) fvalue;
+    case DCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&dvalue);
+	return (FCELL) dvalue;
+    }
+
+    return fvalue;
+}
+
+/*!
+ * \brief Returns the value of type DCELL at position col, row
+ *
+ * The data array can be of type CELL, FCELL or DCELL, the value will be casted to the DCELL type.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \return DCELL
+ *
+ * */
+DCELL N_get_array_2d_d_value(N_array_2d * data, int col, int row)
+{
+    CELL value = 0;
+    FCELL fvalue = 0.0;
+    DCELL dvalue = 0.0;
+
+    switch (data->type) {
+    case CELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&value);
+	return (DCELL) value;
+    case FCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&fvalue);
+	return (DCELL) fvalue;
+    case DCELL_TYPE:
+	N_get_array_2d_value(data, col, row, (void *)&dvalue);
+	return (DCELL) dvalue;
+    }
+
+    return dvalue;
+
+}
+
+/*!
+ * \brief Writes a value to the N_array_2d struct at position col, row
+ *
+ * The value will be automatically cast to the array type.
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \param value char *
+ * \return void
+ * */
+void N_put_array_2d_value(N_array_2d * data, int col, int row, char *value)
+{
+
+    G_debug(6, "N_put_array_2d_value: put value to array");
+
+    if (data->offset == 0) {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    data->cell_array[row * data->cols_intern + col] =
+		*((CELL *) value);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    data->fcell_array[row * data->cols_intern + col] =
+		*((FCELL *) value);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    data->dcell_array[row * data->cols_intern + col] =
+		*((DCELL *) value);
+	}
+    }
+    else {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    data->cell_array[(row + data->offset) * data->cols_intern + col +
+			     data->offset] = *((CELL *) value);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    data->fcell_array[(row + data->offset) * data->cols_intern + col +
+			      data->offset] = *((FCELL *) value);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    data->dcell_array[(row + data->offset) * data->cols_intern + col +
+			      data->offset] = *((DCELL *) value);
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Writes the null value to the N_array_2d struct at position col, row
+ *
+ * The null value will be automatically set to the array data type (CELL, FCELL or DCELL).
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \return void
+ * */
+void N_put_array_2d_value_null(N_array_2d * data, int col, int row)
+{
+
+    G_debug(6,
+	    "N_put_array_2d_value_null: put null value to array pos [%i][%i]",
+	    col, row);
+
+    if (data->offset == 0) {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    Rast_set_c_null_value((void *)
+			       &(data->
+				 cell_array[row * data->cols_intern + col]),
+			       1);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    Rast_set_f_null_value((void *)
+			       &(data->
+				 fcell_array[row * data->cols_intern + col]),
+			       1);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    Rast_set_d_null_value((void *)
+			       &(data->
+				 dcell_array[row * data->cols_intern + col]),
+			       1);
+	}
+    }
+    else {
+	if (data->type == CELL_TYPE && data->cell_array != NULL) {
+	    Rast_set_c_null_value((void *)
+			       &(data->
+				 cell_array[(row +
+					     data->offset) *
+					    data->cols_intern + col +
+					    data->offset]), 1);
+	}
+	else if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    Rast_set_f_null_value((void *)
+			       &(data->
+				 fcell_array[(row +
+					      data->offset) *
+					     data->cols_intern + col +
+					     data->offset]), 1);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    Rast_set_d_null_value((void *)
+			       &(data->
+				 dcell_array[(row +
+					      data->offset) *
+					     data->cols_intern + col +
+					     data->offset]), 1);
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Writes a CELL value to the N_array_2d struct at position col, row
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \param value CELL
+ * \return void
+ * */
+void N_put_array_2d_c_value(N_array_2d * data, int col, int row, CELL value)
+{
+    FCELL fvalue;
+    DCELL dvalue;
+
+    switch (data->type) {
+    case FCELL_TYPE:
+	fvalue = (FCELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&fvalue);
+	return;
+    case DCELL_TYPE:
+	dvalue = (DCELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&dvalue);
+	return;
+    }
+
+    N_put_array_2d_value(data, col, row, (char *)&value);
+
+    return;
+}
+
+/*!
+ * \brief Writes a FCELL value to the N_array_2d struct at position col, row
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \param value FCELL
+ * \return void
+ * */
+void N_put_array_2d_f_value(N_array_2d * data, int col, int row, FCELL value)
+{
+    CELL cvalue;
+    DCELL dvalue;
+
+    switch (data->type) {
+    case CELL_TYPE:
+	cvalue = (CELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&cvalue);
+	return;
+    case DCELL_TYPE:
+	dvalue = (DCELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&dvalue);
+	return;
+    }
+
+    N_put_array_2d_value(data, col, row, (char *)&value);
+
+    return;
+}
+
+/*!
+ * \brief Writes a DCELL value to the N_array_2d struct at position col, row
+ *
+ * \param data N_array_2d *
+ * \param col int
+ * \param row int
+ * \param value DCELL
+ * \return void
+ * */
+void N_put_array_2d_d_value(N_array_2d * data, int col, int row, DCELL value)
+{
+    CELL cvalue;
+    FCELL fvalue;
+
+    switch (data->type) {
+    case CELL_TYPE:
+	cvalue = (CELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&cvalue);
+	return;
+    case FCELL_TYPE:
+	fvalue = (FCELL) value;
+	N_put_array_2d_value(data, col, row, (char *)&fvalue);
+	return;
+    }
+
+    N_put_array_2d_value(data, col, row, (char *)&value);
+
+    return;
+}
+
+/*!
+ * \brief This function writes the data info of the array data to stdout
+ *
+ * \param data N_array_2d *
+ * \return void
+ * */
+void N_print_array_2d_info(N_array_2d * data)
+{
+
+    fprintf(stdout, "N_array_2d \n");
+    fprintf(stdout, "Cols %i\n", data->cols);
+    fprintf(stdout, "Rows: %i\n", data->rows);
+    fprintf(stdout, "Array type: %i\n", data->type);
+    fprintf(stdout, "Offset: %i\n", data->offset);
+    fprintf(stdout, "Internal cols: %i\n", data->cols_intern);
+    fprintf(stdout, "Internal rows: %i\n", data->rows_intern);
+    fprintf(stdout, "CELL array pointer: %p\n", data->cell_array);
+    fprintf(stdout, "FCELL array pointer: %p\n", data->fcell_array);
+    fprintf(stdout, "DCELL array pointer: %p\n", data->dcell_array);
+
+
+    return;
+}
+
+/*!
+ * \brief Write info and content of the N_array_2d struct to stdout
+ *
+ * Offsets are ignored
+ *
+ * \param data N_array_2d *
+ * \return void
+ * */
+void N_print_array_2d(N_array_2d * data)
+{
+    int i, j;
+
+    N_print_array_2d_info(data);
+
+    for (j = 0 - data->offset; j < data->rows + data->offset; j++) {
+	for (i = 0 - data->offset; i < data->cols + data->offset; i++) {
+	    if (data->type == CELL_TYPE)
+		fprintf(stdout, "%6d ", N_get_array_2d_c_value(data, i, j));
+	    else if (data->type == FCELL_TYPE)
+		fprintf(stdout, "%6.6f ", N_get_array_2d_f_value(data, i, j));
+	    else if (data->type == DCELL_TYPE)
+		printf("%6.6f ", N_get_array_2d_d_value(data, i, j));
+	}
+	fprintf(stdout, "\n");
+    }
+    fprintf(stdout, "\n");
+
+    return;
+}
+
+
+/* ******************** 3D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Allocate memory for a N_array_3d data structure.
+ *
+ * This functions allocates an array of type N_array_3d and returns a pointer
+ * to the new allocated memory.
+ * <br><br>
+ * The data type of this array set by "type" must be
+ * FCELL_TYPE or DCELL_TYPE accordingly to the raster3d map data types.
+ * The offsets sets the number of boundary cols, rows and depths.
+ * This option is useful to generate homogeneous Neumann boundary conditions around
+ * an array or to establish overlapping boundaries. The arrays are initialized with 0 by default.
+ * <br><br>
+ * If the offset is greater then 0, negative indices are possible.
+ * The data structure of a array with 3 depths, rows and cols and an offset of 1
+ * will looks like this:
+ *
+ \verbatim
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+
+ 0  0  0  0  0
+ 0  0  1  2  0
+ 0  3  4  5  0
+ 0  6  7  8  0
+ 0  0  0  0  0
+
+ 0  0  0  0  0
+ 0  9 10 11  0
+ 0 12 13 14  0
+ 0 15 16 17  0
+ 0  0  0  0  0
+
+ 0  0  0  0  0
+ 0 18 19 20  0
+ 0 21 22 23  0
+ 0 24 25 26  0
+ 0  0  0  0  0
+
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+ 0  0  0  0  0
+
+ \endverbatim
+
+ The depth counts from the bottom to the top.
+
+ * <br><br>
+ * Internal a one dimensional array is allocated to speed up the memory access.
+ * To access the dimensional array with a three dimensional indexing use the provided
+ * get and put functions.
+ *
+ * \param cols int
+ * \param rows int
+ * \param depths int
+ * \param offset int
+ * \param type int
+ * \return N_array_3d *
+ *
+ * */
+N_array_3d *N_alloc_array_3d(int cols, int rows, int depths, int offset,
+			     int type)
+{
+    N_array_3d *data = NULL;
+
+    if (rows < 1 || cols < 1 || depths < 1)
+	G_fatal_error
+	    ("N_alloc_array_3d: depths, cols and rows should be > 0");
+
+    if (type != DCELL_TYPE && type != FCELL_TYPE)
+	G_fatal_error
+	    ("N_alloc_array_3d: Wrong data type, should be FCELL_TYPE or DCELL_TYPE");
+
+    data = (N_array_3d *) G_calloc(1, sizeof(N_array_3d));
+
+    data->cols = cols;
+    data->rows = rows;
+    data->depths = depths;
+    data->type = type;
+    data->offset = offset;
+    data->rows_intern = rows + 2 * offset;
+    data->cols_intern = cols + 2 * offset;
+    data->depths_intern = depths + 2 * offset;
+    data->fcell_array = NULL;
+    data->dcell_array = NULL;
+
+    if (data->type == FCELL_TYPE) {
+	data->fcell_array =
+	    (float *)G_calloc((size_t) data->depths_intern * data->rows_intern *
+			      data->cols_intern, sizeof(float));
+	G_debug(3,
+		"N_alloc_array_3d: float array allocated rows_intern %i cols_intern %i depths_intern %i offset %i",
+		data->rows_intern, data->cols_intern, data->depths_intern,
+		data->offset = offset);
+    }
+    else if (data->type == DCELL_TYPE) {
+	data->dcell_array =
+	    (double *)G_calloc((size_t) data->depths_intern * data->rows_intern *
+			       data->cols_intern, sizeof(double));
+	G_debug(3,
+		"N_alloc_array_3d: double array allocated rows_intern %i cols_intern %i depths_intern %i offset %i",
+		data->rows_intern, data->cols_intern, data->depths_intern,
+		data->offset = offset);
+    }
+
+    return data;
+}
+
+/*!
+ * \brief Release the memory of a N_array_3d
+ *
+ * \param data N_array_3d *
+ * \return void
+ * */
+void N_free_array_3d(N_array_3d * data)
+{
+
+    if (data != NULL) {
+	G_debug(3, "N_free_array_3d: free N_array_3d");
+
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_free(data->fcell_array);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_free(data->dcell_array);
+	}
+
+	G_free(data);
+	data = NULL;
+
+    }
+
+    return;
+}
+
+/*!
+ * \brief Return the data type of the N_array_3d
+ *
+ * The data type can be FCELL_TYPE and DCELL_TYPE accordingly to the raster map data types.
+ *
+ * \param array N_array_3d *
+ * \return type int -- FCELL_TYPE or DCELL_TYPE
+ * */
+int N_get_array_3d_type(N_array_3d * array)
+{
+    return array->type;
+}
+
+
+/*!
+ * \brief This function writes the value of N_array_3d data at position col, row, depth
+ *        to the variable value
+ *
+ * The value must be from the same type as the array. Otherwise you will risk data losses.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \param value void *
+ * \return void
+ * */
+void
+N_get_array_3d_value(N_array_3d * data, int col, int row, int depth,
+		     void *value)
+{
+
+    if (data->offset == 0) {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    *((float *)value) =
+		data->fcell_array[depth *
+				  (data->rows_intern * data->cols_intern) +
+				  row * data->cols_intern + col];
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    *((double *)value) =
+		data->dcell_array[depth *
+				  (data->rows_intern * data->cols_intern) +
+				  row * data->cols_intern + col];
+	}
+    }
+    else {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    *((float *)value) =
+		data->fcell_array[(depth + data->offset) *
+				  (data->rows_intern * data->cols_intern) +
+				  (row + data->offset) * data->cols_intern +
+				  (col + data->offset)];
+
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    *((double *)value) =
+		data->dcell_array[(depth + data->offset) *
+				  (data->rows_intern * data->cols_intern) +
+				  (row + data->offset) * data->cols_intern +
+				  (col + data->offset)];
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief This function returns 1 if value of N_array_3d data at position col, row, depth
+ * is of type null, otherwise 0
+ *
+ * This function checks automatically the type of the array and checks for the
+ * data type null value.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \return void
+ * */
+int N_is_array_3d_value_null(N_array_3d * data, int col, int row, int depth)
+{
+
+    if (data->offset == 0) {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
+		    depth, row, col);
+	    return Rast3d_is_null_value_num((void *)
+				      &(data->
+					fcell_array[depth *
+						    (data->rows_intern *
+						     data->cols_intern) +
+						    row * data->cols_intern +
+						    col]), FCELL_TYPE);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
+		    depth, row, col);
+	    return Rast3d_is_null_value_num((void *)
+				      &(data->
+					dcell_array[depth *
+						    (data->rows_intern *
+						     data->cols_intern) +
+						    row * data->cols_intern +
+						    col]), DCELL_TYPE);
+	}
+    }
+    else {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
+		    depth, row, col);
+	    return Rast3d_is_null_value_num((void *)
+				      &(data->
+					fcell_array[(depth +
+						     data->offset) *
+						    (data->rows_intern *
+						     data->cols_intern) +
+						    (row + data->offset)
+						    * data->cols_intern +
+						    (col + data->offset)]),
+				      FCELL_TYPE);
+
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    G_debug(6,
+		    "N_is_array_3d_value_null: null value is of type DCELL_TYPE at pos [%i][%i][%i]",
+		    depth, row, col);
+	    return Rast3d_is_null_value_num((void *)
+				      &(data->
+					dcell_array[(depth +
+						     data->offset) *
+						    (data->rows_intern *
+						     data->cols_intern) +
+						    (row +
+						     data->offset) *
+						    data->cols_intern + (col +
+									 data->
+									 offset)]),
+				      DCELL_TYPE);
+	}
+    }
+
+    return 0;
+}
+
+/*!
+ * \brief This function returns the value of type float at position col, row, depth
+ *
+ * The data type can be FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \return float
+ *
+ * */
+float N_get_array_3d_f_value(N_array_3d * data, int col, int row, int depth)
+{
+    float fvalue = 0.0;
+    double dvalue = 0.0;
+
+    switch (data->type) {
+    case FCELL_TYPE:
+	N_get_array_3d_value(data, col, row, depth, (void *)&fvalue);
+	return (float)fvalue;
+    case DCELL_TYPE:
+	N_get_array_3d_value(data, col, row, depth, (void *)&dvalue);
+	return (float)dvalue;
+    }
+
+    return fvalue;
+}
+
+/*!
+ * \brief This function returns the value of type float at position col, row, depth
+ *
+ * The data type can be FCELL_TYPE or DCELL_TYPE accordingly to the raster map data types.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \return double
+ *
+ * */
+double N_get_array_3d_d_value(N_array_3d * data, int col, int row, int depth)
+{
+    float fvalue = 0.0;
+    double dvalue = 0.0;
+
+    switch (data->type) {
+
+    case FCELL_TYPE:
+	N_get_array_3d_value(data, col, row, depth, (void *)&fvalue);
+	return (double)fvalue;
+    case DCELL_TYPE:
+	N_get_array_3d_value(data, col, row, depth, (void *)&dvalue);
+	return (double)dvalue;
+    }
+
+    return dvalue;
+}
+
+/*!
+ * \brief This function writes a value to the N_array_3d data at position col, row, depth
+ *
+ * The value will be automatically cast to the array type.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \param value cahr *
+ * \return void
+ * */
+void
+N_put_array_3d_value(N_array_3d * data, int col, int row, int depth,
+		     char *value)
+{
+
+    G_debug(6, "N_put_array_3d_value: put value to array at pos [%i][%i][%i]",
+	    depth, row, col);
+
+    if (data->offset == 0) {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    data->fcell_array[depth *
+			      (data->rows_intern * data->cols_intern) +
+			      row * data->cols_intern + col]
+		= *((float *)value);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+
+	    data->dcell_array[depth *
+			      (data->rows_intern * data->cols_intern) +
+			      row * data->cols_intern + col]
+		= *((double *)value);
+	}
+    }
+    else {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    data->fcell_array[(depth + data->offset) *
+			      (data->rows_intern * data->cols_intern) + (row +
+									 data->
+									 offset)
+			      * data->cols_intern + (col + data->offset)] =
+		*((float *)value);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    data->dcell_array[(depth + data->offset) *
+			      (data->rows_intern * data->cols_intern) + (row +
+									 data->
+									 offset)
+			      * data->cols_intern + (col + data->offset)] =
+		*((double *)value);
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief This function writes a null value to the N_array_3d data at position col, row, depth
+ *
+ * The null value will be automatically set to the array type.
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \return void
+ * */
+void N_put_array_3d_value_null(N_array_3d * data, int col, int row, int depth)
+{
+
+    G_debug(6,
+	    "N_put_array_3d_value_null: put null value to array at pos [%i][%i][%i]",
+	    depth, row, col);
+
+    if (data->offset == 0) {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    Rast3d_set_null_value((void *)
+			     &(data->
+			       fcell_array[depth *
+					   (data->rows_intern *
+					    data->cols_intern) +
+					   row * data->cols_intern + col]), 1,
+			     FCELL_TYPE);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    Rast3d_set_null_value((void *)
+			     &(data->
+			       dcell_array[depth *
+					   (data->rows_intern *
+					    data->cols_intern) +
+					   row * data->cols_intern + col]), 1,
+			     DCELL_TYPE);
+	}
+    }
+    else {
+	if (data->type == FCELL_TYPE && data->fcell_array != NULL) {
+	    Rast3d_set_null_value((void *)
+			     &(data->
+			       fcell_array[(depth +
+					    data->offset) *
+					   (data->rows_intern *
+					    data->cols_intern) + (row +
+								  data->
+								  offset) *
+					   data->cols_intern + (col +
+								data->
+								offset)]), 1,
+			     FCELL_TYPE);
+	}
+	else if (data->type == DCELL_TYPE && data->dcell_array != NULL) {
+	    Rast3d_set_null_value((void *)
+			     &(data->
+			       dcell_array[(depth +
+					    data->offset) *
+					   (data->rows_intern *
+					    data->cols_intern) + (row +
+								  data->
+								  offset) *
+					   data->cols_intern + (col +
+								data->
+								offset)]), 1,
+			     DCELL_TYPE);
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief This function writes a float value to the N_array_3d data at position col, row, depth
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \param value float
+ * \return void
+ * */
+void
+N_put_array_3d_f_value(N_array_3d * data, int col, int row, int depth,
+		       float value)
+{
+    double dval;
+
+    if (data->type == DCELL_TYPE) {
+	dval = (double)value;
+	N_put_array_3d_value(data, col, row, depth, (void *)&dval);
+    }
+    else {
+	N_put_array_3d_value(data, col, row, depth, (void *)&value);
+    }
+
+    return;
+}
+
+/*!
+ * \brief Writes a double value to the N_array_3d struct at position col, row, depth
+ *
+ * \param data N_array_3d *
+ * \param col int
+ * \param row int
+ * \param depth int
+ * \param value double
+ * \return void
+ * */
+void
+N_put_array_3d_d_value(N_array_3d * data, int col, int row, int depth,
+		       double value)
+{
+    float fval;
+
+    if (data->type == FCELL_TYPE) {
+	fval = (double)value;
+	N_put_array_3d_value(data, col, row, depth, (void *)&fval);
+    }
+    else {
+	N_put_array_3d_value(data, col, row, depth, (void *)&value);
+    }
+
+    return;
+}
+
+/*!
+ * \brief Write the info of the array to stdout
+ *
+ * \param data N_array_3d *
+ * \return void
+ * */
+void N_print_array_3d_info(N_array_3d * data)
+{
+
+    fprintf(stdout, "N_array_3d \n");
+    fprintf(stdout, "Cols %i\n", data->cols);
+    fprintf(stdout, "Rows: %i\n", data->rows);
+    fprintf(stdout, "Depths: %i\n", data->depths);
+    fprintf(stdout, "Array type: %i\n", data->type);
+    fprintf(stdout, "Offset: %i\n", data->offset);
+    fprintf(stdout, "Internal cols: %i\n", data->cols_intern);
+    fprintf(stdout, "Internal rows: %i\n", data->rows_intern);
+    fprintf(stdout, "Internal depths: %i\n", data->depths_intern);
+    fprintf(stdout, "FCELL array pointer: %p\n", data->fcell_array);
+    fprintf(stdout, "DCELL array pointer: %p\n", data->dcell_array);
+
+    return;
+}
+
+/*!
+ * \brief Write info and content of the array data to stdout
+ *
+ * Offsets are ignored
+ *
+ * \param data N_array_2d *
+ * \return void
+ * */
+void N_print_array_3d(N_array_3d * data)
+{
+    int i, j, k;
+
+    N_print_array_3d_info(data);
+
+    for (k = 0; k < data->depths; k++) {
+	for (j = 0; j < data->rows; j++) {
+	    for (i = 0; i < data->cols; i++) {
+		if (data->type == FCELL_TYPE)
+		    printf("%6.6f ", N_get_array_3d_f_value(data, i, j, k));
+		else if (data->type == DCELL_TYPE)
+		    printf("%6.6f ", N_get_array_3d_d_value(data, i, j, k));
+	    }
+	    printf("\n");
+	}
+	printf("\n");
+    }
+    printf("\n");
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_arrays_calc.c (from rev 62429, grass/trunk/lib/gpde/N_arrays_calc.c)
===================================================================
--- grass/trunk/lib/gpde/n_arrays_calc.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_arrays_calc.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,889 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:     	Higher level array managment functions 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+#include <math.h>
+
+#include <grass/N_pde.h>
+#include <grass/raster.h>
+#include <grass/glocale.h>
+
+
+/* ******************** 2D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Copy the source N_array_2d struct to the target N_array_2d struct
+ *
+ * The arrays must have the same size and the same offset.
+ *
+ * The array types can be mixed, the values are automatically casted
+ * and the null values are set accordingly.
+ * <br><br>
+ * If you copy a cell array into a dcell array, the values are casted to dcell and 
+ * the null values are converted from cell-null to dcell-null
+ * <br><br>
+ * This function can be called in a parallel region defined with OpenMP.
+ * The copy loop is parallelize with a openmp for pragma.
+ *
+ * \param source N_array_2d *
+ * \param target N_array_2d *
+ * \return void
+ * */
+void N_copy_array_2d(N_array_2d * source, N_array_2d * target)
+{
+    int i;
+    int null = 0;
+
+#pragma omp single
+    {
+	if (source->cols_intern != target->cols_intern)
+	    G_fatal_error
+		("N_copy_array_2d: the arrays are not of equal size");
+
+	if (source->rows_intern != target->rows_intern)
+	    G_fatal_error
+		("N_copy_array_2d: the arrays are not of equal size");
+
+	G_debug(3,
+		"N_copy_array_2d: copy source array to target array size %i",
+		source->cols_intern * source->rows_intern);
+    }
+
+#pragma omp for
+    for (i = 0; i < source->cols_intern * source->rows_intern; i++) {
+	null = 0;
+	if (source->type == CELL_TYPE) {
+	    if (Rast_is_c_null_value((void *)&source->cell_array[i]))
+		null = 1;
+
+	    if (target->type == CELL_TYPE) {
+		target->cell_array[i] = source->cell_array[i];
+	    }
+	    if (target->type == FCELL_TYPE) {
+		if (null)
+		    Rast_set_f_null_value((void *)&(target->fcell_array[i]), 1);
+		else
+		    target->fcell_array[i] = (FCELL) source->cell_array[i];
+	    }
+	    if (target->type == DCELL_TYPE) {
+		if (null)
+		    Rast_set_d_null_value((void *)&(target->dcell_array[i]), 1);
+		else
+		    target->dcell_array[i] = (DCELL) source->cell_array[i];
+	    }
+
+	}
+	if (source->type == FCELL_TYPE) {
+	    if (Rast_is_f_null_value((void *)&source->fcell_array[i]))
+		null = 1;
+
+	    if (target->type == CELL_TYPE) {
+		if (null)
+		    Rast_set_c_null_value((void *)&(target->cell_array[i]), 1);
+		else
+		    target->cell_array[i] = (CELL) source->fcell_array[i];
+	    }
+	    if (target->type == FCELL_TYPE) {
+		target->fcell_array[i] = source->fcell_array[i];
+	    }
+	    if (target->type == DCELL_TYPE) {
+		if (null)
+		    Rast_set_d_null_value((void *)&(target->dcell_array[i]), 1);
+		else
+		    target->dcell_array[i] = (DCELL) source->fcell_array[i];
+	    }
+	}
+	if (source->type == DCELL_TYPE) {
+	    if (Rast_is_d_null_value((void *)&source->dcell_array[i]))
+		null = 1;
+
+	    if (target->type == CELL_TYPE) {
+		if (null)
+		    Rast_set_c_null_value((void *)&(target->cell_array[i]), 1);
+		else
+		    target->cell_array[i] = (CELL) source->dcell_array[i];
+	    }
+	    if (target->type == FCELL_TYPE) {
+		if (null)
+		    Rast_set_f_null_value((void *)&(target->fcell_array[i]), 1);
+		else
+		    target->fcell_array[i] = (FCELL) source->dcell_array[i];
+	    }
+	    if (target->type == DCELL_TYPE) {
+		target->dcell_array[i] = source->dcell_array[i];
+	    }
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Calculate the norm of the two input arrays
+ *
+ * The norm can be of type N_MAXIMUM_NORM or N_EUKLID_NORM.
+ * All arrays must have equal sizes and offsets.
+ * The complete data array inclusively offsets is used for norm calucaltion.
+ * Only non-null values are used to calcualte the norm.
+ *
+
+ * \param a N_array_2d *
+ * \param b N_array_2d *
+ * \param type the type of the norm -> N_MAXIMUM_NORM, N_EUKLID_NORM
+ * \return double the calculated norm
+ * */
+double N_norm_array_2d(N_array_2d * a, N_array_2d * b, int type)
+{
+    int i = 0;
+    double norm = 0.0, tmp = 0.0;
+    double v1 = 0.0, v2 = 0.0;
+
+    if (a->cols_intern != b->cols_intern)
+	G_fatal_error("N_norm_array_2d: the arrays are not of equal size");
+
+    if (a->rows_intern != b->rows_intern)
+	G_fatal_error("N_norm_array_2d: the arrays are not of equal size");
+
+    G_debug(3, "N_norm_array_2d: norm of a and b size %i",
+	    a->cols_intern * a->rows_intern);
+
+    for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
+	v1 = 0.0;
+	v2 = 0.0;
+
+	if (a->type == CELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(a->cell_array[i])))
+		v1 = (double)a->cell_array[i];
+	}
+	if (a->type == FCELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(a->fcell_array[i])))
+		v1 = (double)a->fcell_array[i];
+	}
+	if (a->type == DCELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(a->dcell_array[i])))
+		v1 = (double)a->dcell_array[i];
+	}
+	if (b->type == CELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(b->cell_array[i])))
+		v2 = (double)b->cell_array[i];
+	}
+	if (b->type == FCELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(b->fcell_array[i])))
+		v2 = (double)b->fcell_array[i];
+	}
+	if (b->type == DCELL_TYPE) {
+	    if (!Rast_is_f_null_value((void *)&(b->dcell_array[i])))
+		v2 = (double)b->dcell_array[i];
+	}
+
+	if (type == N_MAXIMUM_NORM) {
+	    tmp = fabs(v2 - v1);
+	    if ((tmp > norm))
+		norm = tmp;
+	}
+	if (type == N_EUKLID_NORM) {
+	    norm += fabs(v2 - v1);
+	}
+    }
+
+    return norm;
+}
+
+/*!
+ * \brief Calculate basic statistics of the N_array_2d struct 
+ *
+ * Calculates the minimum, maximum, sum and the number of 
+ * non null values. The array offset can be included in the calculation.
+ *
+ * \param a N_array_2d * - input array
+ * \param min double* - variable to store the computed minimum
+ * \param max double* - variable to store the computed maximum
+ * \param sum double* - variable to store the computed sum
+ * \param nonull int* - variable to store the number of non null values
+ * \param withoffset - if 1 include offset values in statistic calculation, 0 otherwise 
+ * \return void
+ * */
+void N_calc_array_2d_stats(N_array_2d * a, double *min, double *max,
+			   double *sum, int *nonull, int withoffset)
+{
+    int i, j;
+    double val;
+
+    *sum = 0.0;
+    *nonull = 0;
+
+    if (withoffset == 1) {
+
+	*min =
+	    (double)N_get_array_2d_d_value(a, 0 - a->offset, 0 - a->offset);
+	*max =
+	    (double)N_get_array_2d_d_value(a, 0 - a->offset, 0 - a->offset);
+
+	for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
+	    for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
+		if (!N_is_array_2d_value_null(a, i, j)) {
+		    val = (double)N_get_array_2d_d_value(a, i, j);
+		    if (*min > val)
+			*min = val;
+		    if (*max < val)
+			*max = val;
+		    *sum += val;
+		    (*nonull)++;
+		}
+	    }
+	}
+    }
+    else {
+
+	*min = (double)N_get_array_2d_d_value(a, 0, 0);
+	*max = (double)N_get_array_2d_d_value(a, 0, 0);
+
+
+	for (j = 0; j < a->rows; j++) {
+	    for (i = 0; i < a->cols; i++) {
+		if (!N_is_array_2d_value_null(a, i, j)) {
+		    val = (double)N_get_array_2d_d_value(a, i, j);
+		    if (*min > val)
+			*min = val;
+		    if (*max < val)
+			*max = val;
+		    *sum += val;
+		    (*nonull)++;
+		}
+	    }
+	}
+    }
+
+    G_debug(3,
+	    "N_calc_array_2d_stats: compute array stats, min %g, max %g, sum %g, nonull %i",
+	    *min, *max, *sum, *nonull);
+    return;
+}
+
+
+/*!
+ * \brief Perform calculations with two input arrays, 
+ * the result is written to a third array.
+ *
+ * All arrays must have equal sizes and offsets.
+ * The complete data array inclusively offsets is used for calucaltions.
+ * Only non-null values are computed. If one array value is null, 
+ * the result array value will be null too.
+ * <br><br>
+ * If a division with zero is detected, the resulting arrays 
+ * value will set to null and not to NaN.
+ * <br><br>
+ * The result array is optional, if the result arrays points to NULL,
+ * a new array will be allocated with the largest arrays data type
+ * (CELL, FCELL or DCELL) used by the input arrays.
+ * <br><br>
+ * the array computations can be of the following forms:
+ *
+ * <ul>
+ * <li>result = a + b -> N_ARRAY_SUM</li>
+ * <li>result = a - b -> N_ARRAY_DIF</li>
+ * <li>result = a * b -> N_ARRAY_MUL</li>
+ * <li>result = a / b -> N_ARRAY_DIV</li>
+ * </ul>
+ *
+ * \param a N_array_2d * - first input array
+ * \param b N_array_2d * - second input array
+ * \param result N_array_2d * - the optional result array
+ * \param type  - the type of calculation
+ * \return N_array_2d * - the pointer to the result array
+ * */
+N_array_2d *N_math_array_2d(N_array_2d * a, N_array_2d * b,
+			    N_array_2d * result, int type)
+{
+    N_array_2d *c;
+    int i, j, setnull = 0;
+    double va = 0.0, vb = 0.0, vc = 0.0;	/*variables used for calculation */
+
+    /*Set the pointer */
+    c = result;
+
+#pragma omp single
+    {
+	/*Check the array sizes */
+	if (a->cols_intern != b->cols_intern)
+	    G_fatal_error
+		("N_math_array_2d: the arrays are not of equal size");
+	if (a->rows_intern != b->rows_intern)
+	    G_fatal_error
+		("N_math_array_2d: the arrays are not of equal size");
+	if (a->offset != b->offset)
+	    G_fatal_error
+		("N_math_array_2d: the arrays have different offsets");
+
+	G_debug(3, "N_math_array_2d: mathematical calculations, size: %i",
+		a->cols_intern * a->rows_intern);
+
+	/*if the result array is null, allocate a new one, use the 
+	 * largest data type of the input arrays*/
+	if (c == NULL) {
+	    if (a->type == DCELL_TYPE || b->type == DCELL_TYPE) {
+		c = N_alloc_array_2d(a->cols, a->rows, a->offset, DCELL_TYPE);
+		G_debug(3,
+			"N_math_array_2d: array of type DCELL_TYPE created");
+	    }
+	    else if (a->type == FCELL_TYPE || b->type == FCELL_TYPE) {
+		c = N_alloc_array_2d(a->cols, a->rows, a->offset, FCELL_TYPE);
+		G_debug(3,
+			"N_math_array_2d: array of type FCELL_TYPE created");
+	    }
+	    else {
+		c = N_alloc_array_2d(a->cols, a->rows, a->offset, CELL_TYPE);
+		G_debug(3,
+			"N_math_array_2d: array of type CELL_TYPE created");
+	    }
+	}
+	else {
+	    /*Check the array sizes */
+	    if (a->cols_intern != c->cols_intern)
+		G_fatal_error
+		    ("N_math_array_2d: the arrays are not of equal size");
+	    if (a->rows_intern != c->rows_intern)
+		G_fatal_error
+		    ("N_math_array_2d: the arrays are not of equal size");
+	    if (a->offset != c->offset)
+		G_fatal_error
+		    ("N_math_array_2d: the arrays have different offsets");
+	}
+    }
+
+#pragma omp for private(va, vb, vc, setnull)
+    for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
+	for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
+	    if (!N_is_array_2d_value_null(a, i, j) &&
+		!N_is_array_2d_value_null(b, i, j)) {
+		/*we always calulate internally with double values */
+		va = (double)N_get_array_2d_d_value(a, i, j);
+		vb = (double)N_get_array_2d_d_value(b, i, j);
+		vc = 0;
+		setnull = 0;
+
+		switch (type) {
+		case N_ARRAY_SUM:
+		    vc = va + vb;
+		    break;
+		case N_ARRAY_DIF:
+		    vc = va - vb;
+		    break;
+		case N_ARRAY_MUL:
+		    vc = va * vb;
+		    break;
+		case N_ARRAY_DIV:
+		    if (vb != 0)
+			vc = va / vb;
+		    else
+			setnull = 1;
+		    break;
+		}
+
+		if (c->type == CELL_TYPE) {
+		    if (setnull)
+			N_put_array_2d_value_null(c, i, j);
+		    else
+			N_put_array_2d_c_value(c, i, j, (CELL) vc);
+		}
+		if (c->type == FCELL_TYPE) {
+		    if (setnull)
+			N_put_array_2d_value_null(c, i, j);
+		    else
+			N_put_array_2d_f_value(c, i, j, (FCELL) vc);
+		}
+		if (c->type == DCELL_TYPE) {
+		    if (setnull)
+			N_put_array_2d_value_null(c, i, j);
+		    else
+			N_put_array_2d_d_value(c, i, j, (DCELL) vc);
+		}
+
+	    }
+	    else {
+		N_put_array_2d_value_null(c, i, j);
+	    }
+	}
+    }
+
+    return c;
+}
+
+/*!
+ * \brief Convert all null values to zero values
+ *
+ * The complete data array inclusively offsets is used.
+ * The array data types are automatically recognized.
+ *
+ * \param a N_array_2d *
+ * \return int - number of replaced values
+ * */
+int N_convert_array_2d_null_to_zero(N_array_2d * a)
+{
+    int i = 0, count = 0;
+
+    G_debug(3, "N_convert_array_2d_null_to_zero: convert array of size %i",
+	    a->cols_intern * a->rows_intern);
+
+    if (a->type == CELL_TYPE)
+	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
+	    if (Rast_is_c_null_value((void *)&(a->cell_array[i]))) {
+		a->cell_array[i] = 0;
+		count++;
+	    }
+	}
+
+    if (a->type == FCELL_TYPE)
+	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
+	    if (Rast_is_f_null_value((void *)&(a->fcell_array[i]))) {
+		a->fcell_array[i] = 0.0;
+		count++;
+	    }
+	}
+
+
+    if (a->type == DCELL_TYPE)
+	for (i = 0; i < a->cols_intern * a->rows_intern; i++) {
+	    if (Rast_is_d_null_value((void *)&(a->dcell_array[i]))) {
+		a->dcell_array[i] = 0.0;
+		count++;
+	    }
+	}
+
+
+    if (a->type == CELL_TYPE)
+	G_debug(2,
+		"N_convert_array_2d_null_to_zero: %i values of type CELL_TYPE are converted",
+		count);
+    if (a->type == FCELL_TYPE)
+	G_debug(2,
+		"N_convert_array_2d_null_to_zero: %i valuess of type FCELL_TYPE are converted",
+		count);
+    if (a->type == DCELL_TYPE)
+	G_debug(2,
+		"N_convert_array_2d_null_to_zero: %i valuess of type DCELL_TYPE are converted",
+		count);
+
+    return count;
+}
+
+/* ******************** 3D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Copy the source N_array_3d struct to the target N_array_3d struct
+ *
+ * The arrays must have the same size and the same offset.
+ *
+ * The array data types can be mixed, the values are automatically casted
+ * and the null values are set accordingly.
+ *
+ * If you copy a float array to a double array, the values are casted to DCELL and 
+ * the null values are converted from FCELL-null to DCELL-null
+ *
+ * \param source N_array_3d *
+ * \param target N_array_3d *
+ * \return void
+ * */
+void N_copy_array_3d(N_array_3d * source, N_array_3d * target)
+{
+    int i;
+    int null;
+
+    if (source->cols_intern != target->cols_intern)
+	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
+
+    if (source->rows_intern != target->rows_intern)
+	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
+
+    if (source->depths_intern != target->depths_intern)
+	G_fatal_error("N_copy_array_3d: the arrays are not of equal size");
+
+
+    G_debug(3, "N_copy_array_3d: copy source array to target array size %i",
+	    source->cols_intern * source->rows_intern *
+	    source->depths_intern);
+
+    for (i = 0;
+	 i <
+	 source->cols_intern * source->rows_intern * source->depths_intern;
+	 i++) {
+	null = 0;
+	if (source->type == FCELL_TYPE) {
+	    if (Rast3d_is_null_value_num
+		((void *)&(source->fcell_array[i]), FCELL_TYPE))
+		null = 1;
+
+	    if (target->type == FCELL_TYPE) {
+		target->fcell_array[i] = source->fcell_array[i];
+	    }
+	    if (target->type == DCELL_TYPE) {
+		if (null)
+		    Rast3d_set_null_value((void *)&(target->dcell_array[i]), 1,
+				     DCELL_TYPE);
+		else
+		    target->dcell_array[i] = (double)source->fcell_array[i];
+	    }
+
+	}
+	if (source->type == DCELL_TYPE) {
+	    if (Rast3d_is_null_value_num
+		((void *)&(source->dcell_array[i]), DCELL_TYPE))
+		null = 1;
+
+	    if (target->type == FCELL_TYPE) {
+		if (null)
+		    Rast3d_set_null_value((void *)&(target->fcell_array[i]), 1,
+				     FCELL_TYPE);
+		else
+		    target->fcell_array[i] = (float)source->dcell_array[i];
+	    }
+	    if (target->type == DCELL_TYPE) {
+		target->dcell_array[i] = source->dcell_array[i];
+	    }
+	}
+    }
+
+    return;
+}
+
+
+/*!
+ * \brief Calculate the norm of the two input arrays
+ *
+ * The norm can be of type N_MAXIMUM_NORM or N_EUKLID_NORM.
+ * All arrays must have equal sizes and offsets.
+ * The complete data array inclusively offsets is used for norm calucaltion.
+ * Only non-null values are used to calcualte the norm.
+ *
+ * \param a N_array_3d *
+ * \param b N_array_3d *
+ * \param type the type of the norm -> N_MAXIMUM_NORM, N_EUKLID_NORM
+ * \return double the calculated norm
+ * */
+double N_norm_array_3d(N_array_3d * a, N_array_3d * b, int type)
+{
+    int i = 0;
+    double norm = 0.0, tmp = 0.0;
+    double v1 = 0.0, v2 = 0.0;
+
+    if (a->cols_intern != b->cols_intern)
+	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
+
+    if (a->rows_intern != b->rows_intern)
+	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
+
+    if (a->depths_intern != b->depths_intern)
+	G_fatal_error("N_norm_array_3d: the arrays are not of equal size");
+
+    G_debug(3, "N_norm_array_3d: norm of a and b size %i",
+	    a->cols_intern * a->rows_intern * a->depths_intern);
+
+    for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern; i++) {
+	v1 = 0.0;
+	v2 = 0.0;
+
+	if (a->type == FCELL_TYPE) {
+	    if (!Rast3d_is_null_value_num((void *)&(a->fcell_array[i]), FCELL_TYPE))
+		v1 = (double)a->fcell_array[i];
+	}
+	if (a->type == DCELL_TYPE) {
+	    if (!Rast3d_is_null_value_num((void *)&(a->dcell_array[i]), DCELL_TYPE))
+		v1 = (double)a->dcell_array[i];
+	}
+	if (b->type == FCELL_TYPE) {
+	    if (!Rast3d_is_null_value_num((void *)&(b->fcell_array[i]), FCELL_TYPE))
+		v2 = (double)b->fcell_array[i];
+	}
+	if (b->type == DCELL_TYPE) {
+	    if (!Rast3d_is_null_value_num((void *)&(b->dcell_array[i]), DCELL_TYPE))
+		v2 = (double)b->dcell_array[i];
+	}
+
+	if (type == N_MAXIMUM_NORM) {
+	    tmp = fabs(v2 - v1);
+	    if ((tmp > norm))
+		norm = tmp;
+	}
+	if (type == N_EUKLID_NORM) {
+	    norm += fabs(v2 - v1);
+	}
+    }
+
+    return norm;
+}
+
+/*!
+ * \brief Calculate basic statistics of the N_array_3d struct
+ *
+ * Calculates the minimum, maximum, sum and the number of 
+ * non null values. The array offset can be included in the statistical calculation.
+ *
+ * \param a N_array_3d * - input array
+ * \param min double* - variable to store the computed minimum
+ * \param max double* - variable to store the computed maximum
+ * \param sum double* - variable to store the computed sum
+ * \param nonull int* - variable to store the number of non null values
+ * \param withoffset - if 1 include offset values in statistic calculation, 0 otherwise 
+ * \return void
+ * */
+void N_calc_array_3d_stats(N_array_3d * a, double *min, double *max,
+			   double *sum, int *nonull, int withoffset)
+{
+    int i, j, k;
+    double val;
+
+    *sum = 0.0;
+    *nonull = 0;
+
+    if (withoffset == 1) {
+
+	*min =
+	    (double)N_get_array_3d_d_value(a, 0 - a->offset, 0 - a->offset,
+					   0 - a->offset);
+	*max =
+	    (double)N_get_array_3d_d_value(a, 0 - a->offset, 0 - a->offset,
+					   0 - a->offset);
+
+	for (k = 0 - a->offset; k < a->depths + a->offset; k++) {
+	    for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
+		for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
+		    if (!N_is_array_3d_value_null(a, i, j, k)) {
+			val = (double)N_get_array_3d_d_value(a, i, j, k);
+			if (*min > val)
+			    *min = val;
+			if (*max < val)
+			    *max = val;
+			*sum += val;
+			(*nonull)++;
+		    }
+		}
+	    }
+	}
+    }
+    else {
+
+	*min = (double)N_get_array_3d_d_value(a, 0, 0, 0);
+	*max = (double)N_get_array_3d_d_value(a, 0, 0, 0);
+
+	for (k = 0; k < a->depths; k++) {
+	    for (j = 0; j < a->rows; j++) {
+		for (i = 0; i < a->cols; i++) {
+		    if (!N_is_array_3d_value_null(a, i, j, k)) {
+			val = (double)N_get_array_3d_d_value(a, i, j, k);
+			if (*min > val)
+			    *min = val;
+			if (*max < val)
+			    *max = val;
+			*sum += val;
+			(*nonull)++;
+		    }
+		}
+	    }
+	}
+    }
+
+    G_debug(3,
+	    "N_calc_array_3d_stats: compute array stats, min %g, max %g, sum %g, nonull %i",
+	    *min, *max, *sum, *nonull);
+
+    return;
+}
+
+/*!
+ * \brief Perform calculations with two input arrays, 
+ * the result is written to a third array.
+ *
+ * All arrays must have equal sizes and offsets.
+ * The complete data array inclusively offsets is used for calucaltions.
+ * Only non-null values are used. If one array value is null, 
+ * the result array value will be null too.
+ * <br><br>
+ *
+ * If a division with zero is detected, the resulting arrays 
+ * value will set to null and not to NaN.
+ * <br><br>
+ *
+ * The result array is optional, if the result arrays points to NULL,
+ * a new array will be allocated with the largest arrays data type
+ * (FCELL_TYPE or DCELL_TYPE) used by the input arrays.
+ * <br><br>
+ *
+ * the calculations are of the following form:
+ *
+ * <ul>
+ * <li>result = a + b -> N_ARRAY_SUM</li>
+ * <li>result = a - b -> N_ARRAY_DIF</li>
+ * <li>result = a * b -> N_ARRAY_MUL</li>
+ * <li>result = a / b -> N_ARRAY_DIV</li>
+ * </ul>
+ *
+ * \param a N_array_3d * - first input array
+ * \param b N_array_3d * - second input array
+ * \param result N_array_3d * - the optional result array
+ * \param type  - the type of calculation
+ * \return N_array_3d * - the pointer to the result array
+ * */
+N_array_3d *N_math_array_3d(N_array_3d * a, N_array_3d * b,
+			    N_array_3d * result, int type)
+{
+    N_array_3d *c;
+    int i, j, k, setnull = 0;
+    double va = 0.0, vb = 0.0, vc = 0.0;	/*variables used for calculation */
+
+    /*Set the pointer */
+    c = result;
+
+    /*Check the array sizes */
+    if (a->cols_intern != b->cols_intern)
+	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
+    if (a->rows_intern != b->rows_intern)
+	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
+    if (a->depths_intern != b->depths_intern)
+	G_fatal_error("N_math_array_3d: the arrays are not of equal size");
+    if (a->offset != b->offset)
+	G_fatal_error("N_math_array_3d: the arrays have different offsets");
+
+    G_debug(3, "N_math_array_3d: mathematical calculations, size: %i",
+	    a->cols_intern * a->rows_intern * a->depths_intern);
+
+    /*if the result array is null, allocate a new one, use the 
+     * largest data type of the input arrays*/
+    if (c == NULL) {
+	if (a->type == DCELL_TYPE || b->type == DCELL_TYPE) {
+	    c = N_alloc_array_3d(a->cols, a->rows, a->depths, a->offset,
+				 DCELL_TYPE);
+	    G_debug(3, "N_math_array_3d: array of type DCELL_TYPE created");
+	}
+	else {
+	    c = N_alloc_array_3d(a->cols, a->rows, a->depths, a->offset,
+				 FCELL_TYPE);
+	    G_debug(3, "N_math_array_3d: array of type FCELL_TYPE created");
+	}
+    }
+    else {
+	/*Check the array sizes */
+	if (a->cols_intern != c->cols_intern)
+	    G_fatal_error
+		("N_math_array_3d: the arrays are not of equal size");
+	if (a->rows_intern != c->rows_intern)
+	    G_fatal_error
+		("N_math_array_3d: the arrays are not of equal size");
+	if (a->depths_intern != c->depths_intern)
+	    G_fatal_error
+		("N_math_array_3d: the arrays are not of equal size");
+	if (a->offset != c->offset)
+	    G_fatal_error
+		("N_math_array_3d: the arrays have different offsets");
+    }
+
+    for (k = 0 - a->offset; k < a->depths + a->offset; k++) {
+	for (j = 0 - a->offset; j < a->rows + a->offset; j++) {
+	    for (i = 0 - a->offset; i < a->cols + a->offset; i++) {
+		if (!N_is_array_3d_value_null(a, i, j, k) &&
+		    !N_is_array_3d_value_null(a, i, j, k)) {
+		    /*we always calulate internally with double values */
+		    va = (double)N_get_array_3d_d_value(a, i, j, k);
+		    vb = (double)N_get_array_3d_d_value(b, i, j, k);
+		    vc = 0;
+		    setnull = 0;
+
+		    switch (type) {
+		    case N_ARRAY_SUM:
+			vc = va + vb;
+			break;
+		    case N_ARRAY_DIF:
+			vc = va - vb;
+			break;
+		    case N_ARRAY_MUL:
+			vc = va * vb;
+			break;
+		    case N_ARRAY_DIV:
+			if (vb != 0)
+			    vc = va / vb;
+			else
+			    setnull = 1;
+			break;
+		    }
+
+		    if (c->type == FCELL_TYPE) {
+			if (setnull)
+			    N_put_array_3d_value_null(c, i, j, k);
+			else
+			    N_put_array_3d_f_value(c, i, j, k, (float)vc);
+		    }
+		    if (c->type == DCELL_TYPE) {
+			if (setnull)
+			    N_put_array_3d_value_null(c, i, j, k);
+			else
+			    N_put_array_3d_d_value(c, i, j, k, vc);
+		    }
+		}
+		else {
+		    N_put_array_3d_value_null(c, i, j, k);
+		}
+	    }
+	}
+    }
+
+    return c;
+}
+
+/*!
+ * \brief Convert all null values to zero values
+ *
+ * The complete data array inclusively offsets is used.
+ *
+ * \param a N_array_3d *
+ * \return int - number of replaced null values
+ * */
+int N_convert_array_3d_null_to_zero(N_array_3d * a)
+{
+    int i = 0, count = 0;
+
+    G_debug(3, "N_convert_array_3d_null_to_zero: convert array of size %i",
+	    a->cols_intern * a->rows_intern * a->depths_intern);
+
+    if (a->type == FCELL_TYPE)
+	for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern;
+	     i++) {
+	    if (Rast3d_is_null_value_num((void *)&(a->fcell_array[i]), FCELL_TYPE)) {
+		a->fcell_array[i] = 0.0;
+		count++;
+	    }
+	}
+
+    if (a->type == DCELL_TYPE)
+	for (i = 0; i < a->cols_intern * a->rows_intern * a->depths_intern;
+	     i++) {
+	    if (Rast3d_is_null_value_num((void *)&(a->dcell_array[i]), DCELL_TYPE)) {
+		a->dcell_array[i] = 0.0;
+		count++;
+	    }
+	}
+
+
+    if (a->type == FCELL_TYPE)
+	G_debug(3,
+		"N_convert_array_3d_null_to_zero: %i values of type FCELL_TYPE are converted",
+		count);
+
+    if (a->type == DCELL_TYPE)
+	G_debug(3,
+		"N_convert_array_3d_null_to_zero: %i values of type DCELL_TYPE are converted",
+		count);
+
+    return count;
+}

Copied: grass/trunk/lib/gpde/n_arrays_io.c (from rev 62429, grass/trunk/lib/gpde/N_arrays_io.c)
===================================================================
--- grass/trunk/lib/gpde/n_arrays_io.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_arrays_io.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,467 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:     	IO array managment functions 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+#include <math.h>
+
+#include <grass/N_pde.h>
+#include <grass/raster.h>
+#include <grass/glocale.h>
+
+
+/* ******************** 2D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Read a raster map into a N_array_2d structure
+ *
+ * The raster map will be opened in the current region settings.
+ * If no N_array_2d structure is provided (NULL pointer), a new structure will be
+ * allocated with the same data type as the raster map and the size of the current region. 
+ * The array offset will be set to 0.
+ * <br><br>
+ * If a N_array_2d structure is provided, the values from the raster map are 
+ * casted to the N_array_2d type. The array must have the same size 
+ * as the current region. 
+ * <br><br>
+ * The new created or the provided array are returned.
+ * If the reading of the raster map fails, G_fatal_error() will
+ * be invoked.
+ *
+ * \param name * char - the name of an existing raster map
+ * \param array * N_array_2d - an existing array or NULL
+ * \return N_array_2d * - the existing or new allocated array
+ * */
+N_array_2d *N_read_rast_to_array_2d(char *name, N_array_2d * array)
+{
+    int map;			/*The rastermap */
+    int x, y, cols, rows, type;
+    void *rast;
+    void *ptr;
+    struct Cell_head region;
+    N_array_2d *data = array;
+
+    /* Get the active region */
+    G_get_set_window(&region);
+
+    /*set the rows and cols */
+    rows = region.rows;
+    cols = region.cols;
+
+    /*open the raster map */
+    map = Rast_open_old(name, "");
+
+    type = Rast_get_map_type(map);
+
+    /*if the array is NULL create a new one with the data type of the raster map */
+    /*the offset is 0 by default */
+    if (data == NULL) {
+	if (type == DCELL_TYPE) {
+	    data = N_alloc_array_2d(cols, rows, 0, DCELL_TYPE);
+	}
+	if (type == FCELL_TYPE) {
+	    data = N_alloc_array_2d(cols, rows, 0, FCELL_TYPE);
+	}
+	if (type == CELL_TYPE) {
+	    data = N_alloc_array_2d(cols, rows, 0, CELL_TYPE);
+	}
+    }
+    else {
+	/*Check the array sizes */
+	if (data->cols != cols)
+	    G_fatal_error
+		("N_read_rast_to_array_2d: the data array size is different from the current region settings");
+	if (data->rows != rows)
+	    G_fatal_error
+		("N_read_rast_to_array_2d: the data array size is different from the current region settings");
+    }
+
+    rast = Rast_allocate_buf(type);
+
+    G_message(_("Reading raster map <%s> into memory"), name);
+
+    for (y = 0; y < rows; y++) {
+	G_percent(y, rows - 1, 10);
+
+	Rast_get_row(map, rast, y, type);
+
+	for (x = 0, ptr = rast; x < cols;
+	     x++, ptr = G_incr_void_ptr(ptr, Rast_cell_size(type))) {
+	    if (type == CELL_TYPE) {
+		if (Rast_is_c_null_value(ptr)) {
+		    N_put_array_2d_value_null(data, x, y);
+		}
+		else {
+		    if (data->type == CELL_TYPE)
+			N_put_array_2d_c_value(data, x, y,
+					       (CELL) * (CELL *) ptr);
+		    if (data->type == FCELL_TYPE)
+			N_put_array_2d_f_value(data, x, y,
+					       (FCELL) * (CELL *) ptr);
+		    if (data->type == DCELL_TYPE)
+			N_put_array_2d_d_value(data, x, y,
+					       (DCELL) * (CELL *) ptr);
+		}
+	    }
+	    if (type == FCELL_TYPE) {
+		if (Rast_is_f_null_value(ptr)) {
+		    N_put_array_2d_value_null(data, x, y);
+		}
+		else {
+		    if (data->type == CELL_TYPE)
+			N_put_array_2d_c_value(data, x, y,
+					       (CELL) * (FCELL *) ptr);
+		    if (data->type == FCELL_TYPE)
+			N_put_array_2d_f_value(data, x, y,
+					       (FCELL) * (FCELL *) ptr);
+		    if (data->type == DCELL_TYPE)
+			N_put_array_2d_d_value(data, x, y,
+					       (DCELL) * (FCELL *) ptr);
+		}
+	    }
+	    if (type == DCELL_TYPE) {
+		if (Rast_is_d_null_value(ptr)) {
+		    N_put_array_2d_value_null(data, x, y);
+		}
+		else {
+		    if (data->type == CELL_TYPE)
+			N_put_array_2d_c_value(data, x, y,
+					       (CELL) * (DCELL *) ptr);
+		    if (data->type == FCELL_TYPE)
+			N_put_array_2d_f_value(data, x, y,
+					       (FCELL) * (DCELL *) ptr);
+		    if (data->type == DCELL_TYPE)
+			N_put_array_2d_d_value(data, x, y,
+					       (DCELL) * (DCELL *) ptr);
+		}
+	    }
+	}
+    }
+
+    /* Close file */
+    Rast_close(map);
+
+    return data;
+}
+
+/*!
+ * \brief Write a N_array_2d struct to a raster map
+ *
+ * A new raster map is created with the same type as the N_array_2d.
+ * The current region is used to open the raster map.
+ * The N_array_2d must have the same size as the current region.
+ If the writing of the raster map fails, G_fatal_error() will
+ * be invoked.
+
+ * \param array N_array_2d * 
+ * \param name char * - the name of the raster map
+ * \return void
+ *
+ * */
+void N_write_array_2d_to_rast(N_array_2d * array, char *name)
+{
+    int map;			/*The rastermap */
+    int x, y, cols, rows, count, type;
+    CELL *rast = NULL;
+    FCELL *frast = NULL;
+    DCELL *drast = NULL;
+    struct Cell_head region;
+
+    if (!array)
+	G_fatal_error(_("N_array_2d * array is empty"));
+
+    /* Get the current region */
+    G_get_set_window(&region);
+
+    rows = region.rows;
+    cols = region.cols;
+    type = array->type;
+
+    /*Open the new map */
+    map = Rast_open_new(name, type);
+
+    if (type == CELL_TYPE)
+	rast = Rast_allocate_buf(type);
+    if (type == FCELL_TYPE)
+	frast = Rast_allocate_buf(type);
+    if (type == DCELL_TYPE)
+	drast = Rast_allocate_buf(type);
+
+    G_message(_("Write 2d array to raster map <%s>"), name);
+
+    count = 0;
+    for (y = 0; y < rows; y++) {
+	G_percent(y, rows - 1, 10);
+	for (x = 0; x < cols; x++) {
+	    if (type == CELL_TYPE)
+		rast[x] = N_get_array_2d_c_value(array, x, y);
+	    if (type == FCELL_TYPE)
+		frast[x] = N_get_array_2d_f_value(array, x, y);
+	    if (type == DCELL_TYPE)
+		drast[x] = N_get_array_2d_d_value(array, x, y);
+	}
+	if (type == CELL_TYPE)
+	    Rast_put_c_row(map, rast);
+	if (type == FCELL_TYPE)
+	    Rast_put_f_row(map, frast);
+	if (type == DCELL_TYPE)
+	    Rast_put_d_row(map, drast);
+    }
+
+    /* Close file */
+    Rast_close(map);
+}
+
+
+/* ******************** 3D ARRAY FUNCTIONS *********************** */
+
+/*!
+ * \brief Read a volume map into a N_array_3d structure
+ *
+ * The volume map is opened in the current region settings.
+ * If no N_array_3d structure is provided (NULL pointer), a new structure will be
+ * allocated with the same data type as the volume map and the size of the current region. 
+ * The array offset will be set to 0.
+ * <br><br>
+ *
+ * If a N_array_3d structure is provided, the values from the volume map are 
+ * casted to the N_array_3d type. The array must have the same size 
+ * as the current region. 
+ * <br><br>
+ *
+ * The new created or the provided array is returned.
+ * If the reading of the volume map fails, Rast3d_fatal_error() will
+ * be invoked.
+ *
+ * \param name * char - the name of an existing volume map
+ * \param array * N_array_3d - an existing array or NULL
+ * \param mask int - 0 = false, 1 = ture : if a mask is presenent, use it with the input volume map
+ * \return N_array_3d * - the existing or new allocated array
+ * */
+N_array_3d *N_read_rast3d_to_array_3d(char *name, N_array_3d * array,
+				      int mask)
+{
+    void *map = NULL;		/*The 3D Rastermap */
+    int changemask = 0;
+    int x, y, z, cols, rows, depths, type;
+    double d1 = 0, f1 = 0;
+    N_array_3d *data = array;
+    RASTER3D_Region region;
+
+
+    /*get the current region */
+    Rast3d_get_window(&region);
+
+    cols = region.cols;
+    rows = region.rows;
+    depths = region.depths;
+
+
+    if (NULL == G_find_raster3d(name, ""))
+	Rast3d_fatal_error(_("3D raster map <%s> not found"), name);
+
+    /*Open all maps with default region */
+    map =
+	Rast3d_open_cell_old(name, G_find_raster3d(name, ""), RASTER3D_DEFAULT_WINDOW,
+			RASTER3D_TILE_SAME_AS_FILE, RASTER3D_USE_CACHE_DEFAULT);
+
+    if (map == NULL)
+	Rast3d_fatal_error(_("Unable to open 3D raster map <%s>"), name);
+
+    type = Rast3d_tile_type_map(map);
+
+    /*if the array is NULL create a new one with the data type of the volume map */
+    /*the offset is 0 by default */
+    if (data == NULL) {
+	if (type == FCELL_TYPE) {
+	    data = N_alloc_array_3d(cols, rows, depths, 0, FCELL_TYPE);
+	}
+	if (type == DCELL_TYPE) {
+	    data = N_alloc_array_3d(cols, rows, depths, 0, DCELL_TYPE);
+	}
+    }
+    else {
+	/*Check the array sizes */
+	if (data->cols != cols)
+	    G_fatal_error
+		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
+	if (data->rows != rows)
+	    G_fatal_error
+		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
+	if (data->depths != depths)
+	    G_fatal_error
+		("N_read_rast_to_array_3d: the data array size is different from the current region settings");
+    }
+
+
+    G_message(_("Read g3d map <%s> into the memory"), name);
+
+    /*if requested set the Mask on */
+    if (mask) {
+	if (Rast3d_mask_file_exists()) {
+	    changemask = 0;
+	    if (Rast3d_mask_is_off(map)) {
+		Rast3d_mask_on(map);
+		changemask = 1;
+	    }
+	}
+    }
+
+    for (z = 0; z < depths; z++) {	/*From the bottom to the top */
+	G_percent(z, depths - 1, 10);
+	for (y = 0; y < rows; y++) {
+	    for (x = 0; x < cols; x++) {
+		if (type == FCELL_TYPE) {
+		    Rast3d_get_value(map, x, y, z, &f1, type);
+		    if (Rast_is_f_null_value((void *)&f1)) {
+			N_put_array_3d_value_null(data, x, y, z);
+		    }
+		    else {
+			if (data->type == FCELL_TYPE)
+			    N_put_array_3d_f_value(data, x, y, z, f1);
+			if (data->type == DCELL_TYPE)
+			    N_put_array_3d_d_value(data, x, y, z, (double)f1);
+		    }
+		}
+		else {
+		    Rast3d_get_value(map, x, y, z, &d1, type);
+		    if (Rast_is_d_null_value((void *)&d1)) {
+			N_put_array_3d_value_null(data, x, y, z);
+		    }
+		    else {
+			if (data->type == FCELL_TYPE)
+			    N_put_array_3d_f_value(data, x, y, z, (float)d1);
+			if (data->type == DCELL_TYPE)
+			    N_put_array_3d_d_value(data, x, y, z, d1);
+		    }
+
+		}
+	    }
+	}
+    }
+
+    /*We set the Mask off, if it was off before */
+    if (mask) {
+	if (Rast3d_mask_file_exists())
+	    if (Rast3d_mask_is_on(map) && changemask)
+		Rast3d_mask_off(map);
+    }
+
+    /* Close files and exit */
+    if (!Rast3d_close(map))
+	Rast3d_fatal_error(map, NULL, 0, _("Error closing g3d file"));
+
+    return data;
+}
+
+/*!
+ * \brief Write a N_array_3d struct to a volume map
+ *
+ * A new volume map is created with the same type as the N_array_3d.
+ * The current region is used to open the volume map.
+ * The N_array_3d must have the same size as the current region.
+ * If the writing of the volume map fails, Rast3d_fatal_error() will
+ * be invoked.
+ *
+ *
+ * \param array N_array_3d * 
+ * \param name char * - the name of the volume map
+ * \param mask int - 1 = use a 3d mask, 0 do not use a 3d mask
+ * \return void
+ *
+ * */
+void N_write_array_3d_to_rast3d(N_array_3d * array, char *name, int mask)
+{
+    void *map = NULL;		/*The 3D Rastermap */
+    int changemask = 0;
+    int x, y, z, cols, rows, depths, count, type;
+    double d1 = 0.0, f1 = 0.0;
+    N_array_3d *data = array;
+    RASTER3D_Region region;
+
+    /*get the current region */
+    Rast3d_get_window(&region);
+
+    cols = region.cols;
+    rows = region.rows;
+    depths = region.depths;
+    type = data->type;
+
+    /*Check the array sizes */
+    if (data->cols != cols)
+	G_fatal_error
+	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
+    if (data->rows != rows)
+	G_fatal_error
+	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
+    if (data->depths != depths)
+	G_fatal_error
+	    ("N_write_array_3d_to_rast3d: the data array size is different from the current region settings");
+
+    /*Open the new map */
+    if (type == DCELL_TYPE)
+        map = Rast3d_open_new_opt_tile_size(name, RASTER3D_USE_CACHE_XY, &region, DCELL_TYPE, 32);
+    else if (type == FCELL_TYPE)
+        map = Rast3d_open_new_opt_tile_size(name, RASTER3D_USE_CACHE_XY, &region, FCELL_TYPE, 32);
+
+    if (map == NULL)
+	Rast3d_fatal_error(_("Error opening g3d map <%s>"), name);
+
+    G_message(_("Write 3d array to g3d map <%s>"), name);
+
+    /*if requested set the Mask on */
+    if (mask) {
+	if (Rast3d_mask_file_exists()) {
+	    changemask = 0;
+	    if (Rast3d_mask_is_off(map)) {
+		Rast3d_mask_on(map);
+		changemask = 1;
+	    }
+	}
+    }
+
+    count = 0;
+    for (z = 0; z < depths; z++) {	/*From the bottom to the top */
+	G_percent(z, depths - 1, 10);
+	for (y = 0; y < rows; y++) {
+	    for (x = 0; x < cols; x++) {
+		if (type == FCELL_TYPE) {
+		    f1 = N_get_array_3d_f_value(data, x, y, z);
+		    Rast3d_put_float(map, x, y, z, f1);
+		}
+		else if (type == DCELL_TYPE) {
+		    d1 = N_get_array_3d_d_value(data, x, y, z);
+		    Rast3d_put_double(map, x, y, z, d1);
+		}
+	    }
+	}
+    }
+
+    /*We set the Mask off, if it was off before */
+    if (mask) {
+	if (Rast3d_mask_file_exists())
+	    if (Rast3d_mask_is_on(map) && changemask)
+		Rast3d_mask_off(map);
+    }
+
+    /* Flush all tile */
+    if (!Rast3d_flush_all_tiles(map))
+	Rast3d_fatal_error("Error flushing tiles with Rast3d_flush_all_tiles");
+    /* Close files and exit */
+    if (!Rast3d_close(map))
+	Rast3d_fatal_error(map, NULL, 0, _("Error closing g3d file"));
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_geom.c (from rev 62429, grass/trunk/lib/gpde/N_geom.c)
===================================================================
--- grass/trunk/lib/gpde/n_geom.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_geom.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,204 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      part of the gpde library
+* 		allocation, destroing and initializing the geometric struct
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+
+#include <grass/N_pde.h>
+
+/* *************************************************************** * 
+ * *********** Konstruktor *************************************** * 
+ * *************************************************************** */
+/*!
+ * \brief Allocate the pde geometry data structure and return a pointer to the new allocated structure
+ *
+ * \return N_geom_data *
+ * */
+N_geom_data *N_alloc_geom_data(void)
+{
+    N_geom_data *geom = (N_geom_data *) G_calloc(1, sizeof(N_geom_data));
+
+    geom->area = NULL;
+    geom->planimetric = 1;
+    geom->dim = 0;
+
+    return geom;
+}
+
+/* *************************************************************** * 
+ * *********** Destruktor **************************************** * 
+ * *************************************************************** */
+/*!
+ * \brief Release memory of a pde geometry data structure
+ *
+ * \param  geom N_geom_data *
+ * \return void
+ * */
+void N_free_geom_data(N_geom_data * geom)
+{
+    if (geom->area != NULL)
+	G_free(geom->area);
+
+    G_free(geom);
+    return;
+}
+
+/* *************************************************************** * 
+ * *************************************************************** * 
+ * *************************************************************** */
+/*!
+ * \brief Initiate a pde geometry data structure with a 3d region
+ *
+ * If the projection is not planimetric, a double array will be created based on the 
+ * number of rows of the provided region
+ *
+ * \param region3d RASTER3D_Region *
+ * \param geodata N_geom_data * - if a NULL pointer is given, a new structure will be allocatet and returned
+ *
+ * \return N_geom_data *
+ * */
+N_geom_data *N_init_geom_data_3d(RASTER3D_Region * region3d, N_geom_data * geodata)
+{
+    N_geom_data *geom = geodata;
+    struct Cell_head region2d;
+
+#pragma omp critical
+    {
+
+	G_debug(2,
+		"N_init_geom_data_3d: initializing the geometry structure");
+
+	if (geom == NULL)
+	    geom = N_alloc_geom_data();
+
+	geom->dz = region3d->tb_res * G_database_units_to_meters_factor();	/*this function is not thread safe */
+	geom->depths = region3d->depths;
+	geom->dim = 3;
+
+	/*convert the 3d into a 2d region and begin the area calculation */
+	G_get_set_window(&region2d);	/*this function is not thread safe */
+	Rast3d_region_to_cell_head(region3d, &region2d);
+    }
+
+    return N_init_geom_data_2d(&region2d, geom);
+}
+
+
+/* *************************************************************** * 
+ * *************************************************************** * 
+ * *************************************************************** */
+/*!
+ * \brief Initiate a pde geometry data structure with a 2d region
+ *
+ * If the projection is not planimetric, a double array will be created based on the 
+ * number of rows of the provided region storing all computed areas for each row
+ *
+ * \param region sruct Cell_head *
+ * \param geodata N_geom_data * - if a NULL pointer is given, a new structure will be allocatet and returned
+ *
+ * \return N_geom_data *
+ * */
+N_geom_data *N_init_geom_data_2d(struct Cell_head * region,
+				 N_geom_data * geodata)
+{
+    N_geom_data *geom = geodata;
+    struct Cell_head backup;
+    double meters;
+    short ll = 0;
+    int i;
+
+
+    /*create an openmp lock to assure that only one thread at a time will access this function */
+#pragma omp critical
+    {
+	G_debug(2,
+		"N_init_geom_data_2d: initializing the geometry structure");
+
+	/*make a backup from this region */
+	G_get_set_window(&backup);	/*this function is not thread safe */
+	/*set the current region */
+	Rast_set_window(region);	/*this function is not thread safe */
+
+	if (geom == NULL)
+	    geom = N_alloc_geom_data();
+
+	meters = G_database_units_to_meters_factor();	/*this function is not thread safe */
+
+	/*set the dim to 2d if it was not initiated with 3, thats a bit ugly :( */
+	if (geom->dim != 3)
+	    geom->dim = 2;
+
+	geom->planimetric = 1;
+	geom->rows = region->rows;
+	geom->cols = region->cols;
+	geom->dx = region->ew_res * meters;
+	geom->dy = region->ns_res * meters;
+	geom->Az = geom->dy * geom->dx;	/*square meters in planimetric proj */
+	/*depths and dz are initialized with a 3d region */
+
+	/*Begin the area calculation */
+	ll = G_begin_cell_area_calculations();	/*this function is not thread safe */
+
+	/*if the projection is not planimetric, calc the area for each row */
+	if (ll == 2) {
+	    G_debug(2,
+		    "N_init_geom_data_2d: calculating the areas for non parametric projection");
+	    geom->planimetric = 0;
+
+	    if (geom->area != NULL)
+		G_free(geom->area);
+	    else
+		geom->area = G_calloc(geom->rows, sizeof(double));
+
+	    /*fill the area vector */
+	    for (i = 0; i < geom->rows; i++) {
+		geom->area[i] = G_area_of_cell_at_row(i);	/*square meters */
+	    }
+	}
+
+	/*restore the old region */
+	Rast_set_window(&backup);	/*this function is not thread safe */
+    }
+
+    return geom;
+}
+
+/* *************************************************************** * 
+ * *************************************************************** * 
+ * *************************************************************** */
+/*!
+ * \brief Get the areay size in square meter of one cell (x*y) at row
+ *
+ * This function works for two and three dimensions
+ *
+ * \param geom N_geom_data *
+ * \param row int
+ * \return area double
+ *
+ * */
+double N_get_geom_data_area_of_cell(N_geom_data * geom, int row)
+{
+    if (geom->planimetric) {
+	G_debug(6, "N_get_geom_data_area_of_cell: %g", geom->Az);
+	return geom->Az;
+    }
+    else {
+	G_debug(6, "N_get_geom_data_area_of_cell: %g", geom->area[row]);
+	return geom->area[row];
+    }
+
+    return 0.0;
+}

Copied: grass/trunk/lib/gpde/n_gradient.c (from rev 62429, grass/trunk/lib/gpde/N_gradient.c)
===================================================================
--- grass/trunk/lib/gpde/n_gradient.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_gradient.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,1113 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:     	gradient management functions 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <grass/N_pde.h>
+
+/*!
+ * \brief Allocate a N_gradient_2d structure
+ *
+ * \return N_gradient_2d *
+ *
+ * */
+N_gradient_2d *N_alloc_gradient_2d(void)
+{
+    N_gradient_2d *grad;
+
+    grad = (N_gradient_2d *) G_calloc(1, sizeof(N_gradient_2d));
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_2d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_2d(N_gradient_2d * grad)
+{
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+/*!
+ * \brief allocate and initialize a N_gradient_2d structure
+ *
+ * \param NC double - the gradient between northern and center cell
+ * \param SC double - the gradient between southern and center cell
+ * \param WC double - the gradient between western and center cell
+ * \param EC double - the gradient between eastern and center cell
+ * \return N_gradient_2d *
+ *
+ * */
+N_gradient_2d *N_create_gradient_2d(double NC, double SC, double WC,
+				    double EC)
+{
+    N_gradient_2d *grad;
+
+    G_debug(5, "N_create_gradient_2d: create N_gradient_2d");
+
+    grad = N_alloc_gradient_2d();
+
+    grad->NC = NC;
+    grad->SC = SC;
+    grad->WC = WC;
+    grad->EC = EC;
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_2d structure
+ *
+ * \param source - the source N_gradient_2d struct
+ * \param target - the target N_gradient_2d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int N_copy_gradient_2d(N_gradient_2d * source, N_gradient_2d * target)
+{
+    G_debug(5, "N_copy_gradient_2d: copy N_gradient_2d");
+
+    if (!source || !target)
+	return 0;
+
+    target->NC = source->NC;
+    target->SC = source->SC;
+    target->WC = source->WC;
+    target->EC = source->EC;
+
+    return 1;
+}
+
+/*!
+ * \brief Return a N_gradient_2d structure calculated from the input gradient field
+ * at position [row][col]
+ *
+ *  This function returns the gradient of a cell at position [row][col] from the input gradient field.
+ *  Returend is a new structure of type N_gradient_2d.
+ *
+ *  \param field N_gradient_field_2d * - A two dimensional gradient field
+ *  \param gradient N_gradient_2d * - the gradient structure which should be filled with data, if a NULL pointer is given, a new structure will be created
+ *  \param col int
+ *  \param row int
+ *  \return N_gradient_2d * - the new or filled gradient structure
+ *  
+ *
+ * */
+N_gradient_2d *N_get_gradient_2d(N_gradient_field_2d * field,
+				 N_gradient_2d * gradient, int col, int row)
+{
+    double NC = 0, SC = 0, WC = 0, EC = 0;
+    N_gradient_2d *grad = gradient;
+
+
+    NC = N_get_array_2d_d_value(field->y_array, col, row);
+    SC = N_get_array_2d_d_value(field->y_array, col, row + 1);
+    WC = N_get_array_2d_d_value(field->x_array, col, row);
+    EC = N_get_array_2d_d_value(field->x_array, col + 1, row);
+
+    G_debug(5,
+	    "N_get_gradient_2d: calculate N_gradient_2d NC %g SC %g WC %g EC %g",
+	    NC, SC, WC, EC);
+
+    /*if gradient is a NULL pointer, create a new one */
+    if (!grad) {
+	grad = N_create_gradient_2d(NC, SC, WC, EC);
+    }
+    else {
+	grad->NC = NC;
+	grad->SC = SC;
+	grad->WC = WC;
+	grad->EC = EC;
+    }
+
+    return grad;
+}
+
+/*!
+ * \brief Allocate a N_gradient_3d structure
+ *
+ * \return N_gradient_3d *
+ *
+ * */
+N_gradient_3d *N_alloc_gradient_3d(void)
+{
+    N_gradient_3d *grad;
+
+    grad = (N_gradient_3d *) G_calloc(1, sizeof(N_gradient_3d));
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_3d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_3d(N_gradient_3d * grad)
+{
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+
+/*!
+ * \brief allocate and initialize a N_gradient_3d structure
+ *
+ * \param NC double - the gradient between northern and center cell
+ * \param SC double - the gradient between southern and center cell
+ * \param WC double - the gradient between western and center cell
+ * \param EC double - the gradient between eastern and center cell
+ * \param TC double - the gradient between top and center cell
+ * \param BC double - the gradient between bottom and center cell
+ * \return N_gradient_3d *
+ *
+ * */
+N_gradient_3d *N_create_gradient_3d(double NC, double SC, double WC,
+				    double EC, double TC, double BC)
+{
+    N_gradient_3d *grad;
+
+    G_debug(5, "N_create_gradient_3d: create N_gradient_3d");
+
+    grad = N_alloc_gradient_3d();
+
+    grad->NC = NC;
+    grad->SC = SC;
+    grad->WC = WC;
+    grad->EC = EC;
+    grad->TC = TC;
+    grad->BC = BC;
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_3d structure
+ *
+ * \param source - the source N_gradient_3d struct
+ * \param target - the target N_gradient_3d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int N_copy_gradient_3d(N_gradient_3d * source, N_gradient_3d * target)
+{
+    G_debug(5, "N_copy_gradient_3d: copy N_gradient_3d");
+
+    if (!source || !target)
+	return 0;
+
+    target->NC = source->NC;
+    target->SC = source->SC;
+    target->WC = source->WC;
+    target->EC = source->EC;
+    target->TC = source->TC;
+    target->BC = source->BC;
+
+    return 1;
+}
+
+
+/*!
+ * \brief Return a N_gradient_3d structure calculated from the input gradient field
+ * at position [depth][row][col]
+ *
+ *  This function returns the gradient of a 3d cell at position [depth][row][col] from the input gradient field.
+ *  Returned is a new structure of type N_gradient_3d.
+ *
+ *  \param field N_gradient_field_3d * - A three dimensional gradient field
+ *  \param gradient N_gradient_3d * - an existing gradient structure or a NULL pointer, if a NULL pointer is providet a new structure will be returned
+ *  \param col int
+ *  \param row int
+ *  \param depth int
+ *  \return N_gradient_3d *
+ *  
+ *
+ * */
+N_gradient_3d *N_get_gradient_3d(N_gradient_field_3d * field,
+				 N_gradient_3d * gradient, int col, int row,
+				 int depth)
+{
+    double NC, SC, WC, EC, TC, BC;
+    N_gradient_3d *grad = gradient;
+
+    NC = N_get_array_3d_d_value(field->y_array, col, row, depth);
+    SC = N_get_array_3d_d_value(field->y_array, col, row + 1, depth);
+    WC = N_get_array_3d_d_value(field->x_array, col, row, depth);
+    EC = N_get_array_3d_d_value(field->x_array, col + 1, row, depth);
+    BC = N_get_array_3d_d_value(field->z_array, col, row, depth);
+    TC = N_get_array_3d_d_value(field->z_array, col, row, depth + 1);
+
+    G_debug(6,
+	    "N_get_gradient_3d: calculate N_gradient_3d NC %g SC %g WC %g EC %g TC %g BC %g",
+	    NC, SC, WC, EC, TC, BC);
+
+    /*if gradient is a NULL pointer, create a new one */
+    if (!grad) {
+	grad = N_create_gradient_3d(NC, SC, WC, EC, TC, BC);
+    }
+    else {
+	grad->NC = NC;
+	grad->SC = SC;
+	grad->WC = WC;
+	grad->EC = EC;
+	grad->BC = BC;
+	grad->TC = TC;
+    }
+
+    return grad;
+}
+
+/*!
+ * \brief Allocate a N_gradient_neighbours_x structure
+ *
+ * This structure contains all neighbour gradients in x direction of one cell  
+ *
+ * \return N_gradient_neighbours_x  *
+ *
+ * */
+N_gradient_neighbours_x *N_alloc_gradient_neighbours_x(void)
+{
+    N_gradient_neighbours_x *grad;
+
+    grad =
+	(N_gradient_neighbours_x *) G_calloc(1,
+					     sizeof(N_gradient_neighbours_x));
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_x structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_neighbours_x(N_gradient_neighbours_x * grad)
+{
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+
+/*!
+ * \brief Allocate and initialize a N_gradient_neighbours_x structure
+ *
+ * \param NWN double - the gradient between north-west and northern cell
+ * \param NEN double - the gradient between north-east and northern cell
+ * \param WC double - the gradient between western and center cell
+ * \param EC double - the gradient between eastern and center cell
+ * \param SWS double - the gradient between south-west and southern cell
+ * \param SES double - the gradient between south-east and southern cell
+ * \return N_gradient_neighbours_x *
+
+ *
+ * */
+N_gradient_neighbours_x *N_create_gradient_neighbours_x(double NWN,
+							double NEN, double WC,
+							double EC, double SWS,
+							double SES)
+{
+    N_gradient_neighbours_x *grad;
+
+    G_debug(6,
+	    "N_create_gradient_neighbours_x: create N_gradient_neighbours_x");
+
+    grad = N_alloc_gradient_neighbours_x();
+
+    grad->NWN = NWN;
+    grad->NEN = NEN;
+    grad->WC = WC;
+    grad->EC = EC;
+    grad->SWS = SWS;
+    grad->SES = SES;
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_neighbours_x structure
+ *
+ * \param source - the source N_gradient_neighbours_x struct
+ * \param target - the target N_gradient_neighbours_x struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_neighbours_x(N_gradient_neighbours_x * source,
+			     N_gradient_neighbours_x * target)
+{
+    G_debug(6, "N_copy_gradient_neighbours_x: copy N_gradient_neighbours_x");
+
+    if (!source || !target)
+	return 0;
+
+    target->NWN = source->NWN;
+    target->NEN = source->NEN;
+    target->WC = source->WC;
+    target->EC = source->EC;
+    target->SWS = source->SWS;
+    target->SES = source->SES;
+
+    return 1;
+}
+
+/*!
+ * \brief Allocate a N_gradient_neighbours_y structure
+ *
+ * This structure contains all neighbour gradients in y direction of one cell  
+ *
+ * \return N_gradient_neighbours_y  *
+ *
+ * */
+N_gradient_neighbours_y *N_alloc_gradient_neighbours_y(void)
+{
+    N_gradient_neighbours_y *grad;
+
+    grad =
+	(N_gradient_neighbours_y *) G_calloc(1,
+					     sizeof(N_gradient_neighbours_y));
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_y structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_neighbours_y(N_gradient_neighbours_y * grad)
+{
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Allocate and initialize a N_gradient_neighbours_y structure
+ *
+ * \param NWW double - the gradient between north-west and western cell
+ * \param NEE double - the gradient between north-east and eastern cell
+ * \param NC double - the gradient between northern and center cell
+ * \param SC double - the gradient between southern and center cell
+ * \param SWW double - the gradient between south-west and western cell
+ * \param SEE double - the gradient between south-east and eastern cell
+ * \return N_gradient_neighbours_y *
+
+ *
+ * */
+N_gradient_neighbours_y *N_create_gradient_neighbours_y(double NWW,
+							double NEE, double NC,
+							double SC, double SWW,
+							double SEE)
+{
+    N_gradient_neighbours_y *grad;
+
+    G_debug(6,
+	    "N_create_gradient_neighbours_y: create N_gradient_neighbours_y");
+
+    grad = N_alloc_gradient_neighbours_y();
+
+    grad->NWW = NWW;
+    grad->NEE = NEE;
+    grad->NC = NC;
+    grad->SC = SC;
+    grad->SWW = SWW;
+    grad->SEE = SEE;
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_neighbours_y structure
+ *
+ * \param source - the source N_gradient_neighbours_y struct
+ * \param target - the target N_gradient_neighbours_y struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_neighbours_y(N_gradient_neighbours_y * source,
+			     N_gradient_neighbours_y * target)
+{
+    G_debug(6, "N_copy_gradient_neighbours_y: copy N_gradient_neighbours_y");
+
+    if (!source || !target)
+	return 0;
+
+    target->NWW = source->NWW;
+    target->NEE = source->NEE;
+    target->NC = source->NC;
+    target->SC = source->SC;
+    target->SWW = source->SWW;
+    target->SEE = source->SEE;
+
+    return 1;
+}
+
+/*!
+ * \brief Allocate a N_gradient_neighbours_z structure
+ *
+ * This structure contains all neighbour gradients in z direction of one cell  
+ *
+ * \return N_gradient_neighbours_z  *
+ *
+ * */
+N_gradient_neighbours_z *N_alloc_gradient_neighbours_z(void)
+{
+    N_gradient_neighbours_z *grad;
+
+    grad =
+	(N_gradient_neighbours_z *) G_calloc(1,
+					     sizeof(N_gradient_neighbours_z));
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_z structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_neighbours_z(N_gradient_neighbours_z * grad)
+{
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Allocate and initialize a N_gradient_neighbours_z structure
+ *
+ * \param NWZ double - the gradient between upper and lower north-western cells
+ * \param NZ double - the gradient between upper and lower northern cells
+ * \param NEZ double - the gradient between upper and lower north-eastern cells
+ * \param WZ double - the gradient between upper and lower western cells
+ * \param CZ double - the gradient between upper and lower center cells
+ * \param EZ double - the gradient between upper and lower eastern cells
+ * \param SWZ double - the gradient between upper and lower south-western cells
+ * \param SZ double - the gradient between upper and lower southern cells
+ * \param SEZ double - the gradient between upper and lower south-eastern cells
+ * \return N_gradient_neighbours_z *
+
+ *
+ * */
+N_gradient_neighbours_z *N_create_gradient_neighbours_z(double NWZ, double NZ,
+							double NEZ, double WZ,
+							double CZ, double EZ,
+							double SWZ, double SZ,
+							double SEZ)
+{
+    N_gradient_neighbours_z *grad;
+
+    G_debug(6,
+	    "N_create_gradient_neighbours_z: create N_gradient_neighbours_z");
+
+    grad = N_alloc_gradient_neighbours_z();
+
+    grad->NWZ = NWZ;
+    grad->NZ = NZ;
+    grad->NEZ = NEZ;
+    grad->WZ = WZ;
+    grad->CZ = CZ;
+    grad->EZ = EZ;
+    grad->SWZ = SWZ;
+    grad->SZ = SZ;
+    grad->SEZ = SEZ;
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_neighbours_z structure
+ *
+ * \param source - the source N_gradient_neighbours_z struct
+ * \param target - the target N_gradient_neighbours_z struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_neighbours_z(N_gradient_neighbours_z * source,
+			     N_gradient_neighbours_z * target)
+{
+    G_debug(6, "N_copy_gradient_neighbours_z: copy N_gradient_neighbours_z");
+
+    if (!source || !target)
+	return 0;
+
+    target->NWZ = source->NWZ;
+    target->NZ = source->NZ;
+    target->NEZ = source->NEZ;
+    target->WZ = source->WZ;
+    target->CZ = source->CZ;
+    target->EZ = source->EZ;
+    target->SWZ = source->SWZ;
+    target->SZ = source->SZ;
+    target->SEZ = source->SEZ;
+
+    return 1;
+}
+
+/*!
+ * \brief Allocate a N_gradient_neighbours_2d structure
+ *
+ * This structure contains all neighbour gradients in all directions of one cell 
+ * in a 2d raster layer
+ *
+ * \return N_gradient_neighbours_2d *
+ *
+ * */
+N_gradient_neighbours_2d *N_alloc_gradient_neighbours_2d(void)
+{
+    N_gradient_neighbours_2d *grad;
+
+    grad =
+	(N_gradient_neighbours_2d *) G_calloc(1,
+					      sizeof
+					      (N_gradient_neighbours_2d));
+
+    grad->x = N_alloc_gradient_neighbours_x();
+    grad->y = N_alloc_gradient_neighbours_y();
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_2d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_neighbours_2d(N_gradient_neighbours_2d * grad)
+{
+
+    N_free_gradient_neighbours_x(grad->x);
+    N_free_gradient_neighbours_y(grad->y);
+
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Allocate and initialize a N_gradient_neighbours_2d structure
+ *
+ * The parameter N_gradient_neighbours x and y are copied into the new allocated structure 
+ * and can be deleted after the initializing
+ *
+ * \return N_gradient_neighbours_2d * -- if failure NULL is returned
+ *
+ * */
+N_gradient_neighbours_2d
+    * N_create_gradient_neighbours_2d(N_gradient_neighbours_x * x,
+				      N_gradient_neighbours_y * y)
+{
+    N_gradient_neighbours_2d *grad;
+    int fail = 0;
+
+    G_debug(5,
+	    "N_create_gradient_neighbours_2d: create N_gradient_neighbours_2d");
+
+    grad = N_alloc_gradient_neighbours_2d();
+
+    if (!N_copy_gradient_neighbours_x(x, grad->x))
+	fail++;
+    if (!N_copy_gradient_neighbours_y(y, grad->y))
+	fail++;
+
+    if (fail > 0) {
+	N_free_gradient_neighbours_2d(grad);
+	grad = NULL;
+    }
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_neighbours_2d structure
+ *
+ * \param source - the source N_gradient_neighbours_2d struct
+ * \param target - the target N_gradient_neighbours_2d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_neighbours_2d(N_gradient_neighbours_2d * source,
+			      N_gradient_neighbours_2d * target)
+{
+    int fail = 0;
+
+    G_debug(5,
+	    "N_copy_gradient_neighbours_2d: copy N_gradient_neighbours_2d");
+
+    if (!source || !target)
+	return 0;
+
+    if (!(N_copy_gradient_neighbours_x(source->x, target->x)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(source->y, target->y)))
+	fail++;
+
+    if (fail > 0) {
+	return 0;
+    }
+
+    return 1;
+}
+
+/*!
+ * \brief Return a N_gradient_neighbours_2d structure calculated from the input gradient field
+ * at position [row][col]
+ *
+ *  This function returns the gradient neighbours in x and y dierection 
+ *  of a cell at position [row][col] from the input gradient field.
+ *  Returend is a pointer to a structure of type N_gradient_neighbours_2d.
+ *
+ *  \param field N_gradient_field_2d * - A two dimensional gradient field
+ *  \param gradient N_gradient_neighbours_2d * - the gradient structure which should be filled with data, if a NULL pointer is given, a new structure will be created
+ *  \param col int
+ *  \param row int
+ *  \return N_gradient_neighbours_2d * - the new or filled gradient structure
+ *  
+ *
+ * */
+N_gradient_neighbours_2d *N_get_gradient_neighbours_2d(N_gradient_field_2d *
+						       field,
+						       N_gradient_neighbours_2d
+						       * gradient, int col,
+						       int row)
+{
+    double NWN, NEN, WC, EC, SWS, SES;
+    double NWW, NEE, NC, SC, SWW, SEE;
+    N_gradient_neighbours_2d *grad = NULL;
+    N_gradient_neighbours_x *grad_x = NULL;
+    N_gradient_neighbours_y *grad_y = NULL;
+
+
+    NWN = N_get_array_2d_d_value(field->x_array, col, row - 1);
+    NEN = N_get_array_2d_d_value(field->x_array, col + 1, row - 1);
+    WC = N_get_array_2d_d_value(field->x_array, col, row);
+    EC = N_get_array_2d_d_value(field->x_array, col + 1, row);
+    SWS = N_get_array_2d_d_value(field->x_array, col, row + 1);
+    SES = N_get_array_2d_d_value(field->x_array, col + 1, row + 1);
+
+    NWW = N_get_array_2d_d_value(field->y_array, col - 1, row);
+    NEE = N_get_array_2d_d_value(field->y_array, col + 1, row);
+    NC = N_get_array_2d_d_value(field->y_array, col, row);
+    SC = N_get_array_2d_d_value(field->y_array, col, row + 1);
+    SWW = N_get_array_2d_d_value(field->y_array, col - 1, row + 1);
+    SEE = N_get_array_2d_d_value(field->y_array, col + 1, row + 1);
+
+
+    grad_x = N_create_gradient_neighbours_x(NWN, NEN, WC, EC, SWS, SES);
+    grad_y = N_create_gradient_neighbours_y(NWW, NEE, NC, SC, SWW, SEE);
+
+    G_debug(5,
+	    "N_get_gradient_neighbours_2d: calculate N_gradient_neighbours_x NWN %g NEN %g WC %g EC %g SWS %g SES %g",
+	    NWN, NEN, WC, EC, SWS, SES);
+
+    G_debug(5,
+	    "N_get_gradient_neighbours_2d: calculate N_gradient_neighbours_y NWW %g NEE %g NC %g SC %g SWW %g SEE %g",
+	    NWW, NEE, NC, SC, SWW, SEE);
+
+
+    /*if gradient is a NULL pointer, create a new one */
+    if (!gradient) {
+	grad = N_create_gradient_neighbours_2d(grad_x, grad_y);
+	gradient = grad;
+    }
+    else {
+	grad = N_create_gradient_neighbours_2d(grad_x, grad_y);
+	N_copy_gradient_neighbours_2d(grad, gradient);
+	N_free_gradient_neighbours_2d(grad);
+    }
+
+    N_free_gradient_neighbours_x(grad_x);
+    N_free_gradient_neighbours_y(grad_y);
+
+    return gradient;
+}
+
+
+/*!
+ * \brief Allocate a N_gradient_neighbours_3d structure
+ *
+ * This structure contains all neighbour gradients in all directions of one cell 
+ * in a 3d raster layer
+ *
+ * \return N_gradient_neighbours_3d *
+ *
+ * */
+N_gradient_neighbours_3d *N_alloc_gradient_neighbours_3d(void)
+{
+    N_gradient_neighbours_3d *grad;
+
+    grad =
+	(N_gradient_neighbours_3d *) G_calloc(1,
+					      sizeof
+					      (N_gradient_neighbours_3d));
+
+    grad->xt = N_alloc_gradient_neighbours_x();
+    grad->xc = N_alloc_gradient_neighbours_x();
+    grad->xb = N_alloc_gradient_neighbours_x();
+    grad->yt = N_alloc_gradient_neighbours_y();
+    grad->yc = N_alloc_gradient_neighbours_y();
+    grad->yb = N_alloc_gradient_neighbours_y();
+    grad->zt = N_alloc_gradient_neighbours_z();
+    grad->zb = N_alloc_gradient_neighbours_z();
+
+    return grad;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_3d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_neighbours_3d(N_gradient_neighbours_3d * grad)
+{
+
+    N_free_gradient_neighbours_x(grad->xt);
+    N_free_gradient_neighbours_x(grad->xc);
+    N_free_gradient_neighbours_x(grad->xb);
+    N_free_gradient_neighbours_y(grad->yt);
+    N_free_gradient_neighbours_y(grad->yc);
+    N_free_gradient_neighbours_y(grad->yb);
+    N_free_gradient_neighbours_z(grad->zt);
+    N_free_gradient_neighbours_z(grad->zb);
+
+    G_free(grad);
+    grad = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Allocate and initialize a N_gradient_neighbours_3d structure
+ *
+ * The parameter N_gradient_neighbours x(tcb) and y(tcb) and z(tb) are copied into the new allocated structure 
+ * and can be deleted after the initializing
+ *
+ * \return N_gradient_neighbours_3d * -- if failure NULL is returned
+
+ *
+ * */
+N_gradient_neighbours_3d
+    * N_create_gradient_neighbours_3d(N_gradient_neighbours_x * xt,
+				      N_gradient_neighbours_x * xc,
+				      N_gradient_neighbours_x * xb,
+				      N_gradient_neighbours_y * yt,
+				      N_gradient_neighbours_y * yc,
+				      N_gradient_neighbours_y * yb,
+				      N_gradient_neighbours_z * zt,
+				      N_gradient_neighbours_z * zb)
+{
+    N_gradient_neighbours_3d *grad;
+    int fail = 0;
+
+    G_debug(5,
+	    "N_create_gradient_neighbours_3d: create N_gradient_neighbours_3d");
+
+    grad = N_alloc_gradient_neighbours_3d();
+
+    if (!(N_copy_gradient_neighbours_x(xt, grad->xt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_x(xc, grad->xc)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_x(xb, grad->xb)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(yt, grad->yt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(yc, grad->yc)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(yb, grad->yb)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_z(zt, grad->zt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_z(zb, grad->zb)))
+	fail++;
+
+    if (fail > 0) {
+	return NULL;
+    }
+
+    return grad;
+}
+
+/*!
+ * \brief copy a N_gradient_neighbours_3d structure
+ *
+ * \param source - the source N_gradient_neighbours_3d struct
+ * \param target - the target N_gradient_neighbours_3d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_neighbours_3d(N_gradient_neighbours_3d * source,
+			      N_gradient_neighbours_3d * target)
+{
+    int fail = 0;
+
+    G_debug(5,
+	    "N_copy_gradient_neighbours_3d: copy N_gradient_neighbours_3d");
+
+    if (!source || !target)
+	return 0;
+
+    if (!(N_copy_gradient_neighbours_x(source->xt, target->xt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_x(source->xc, target->xc)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_x(source->xb, target->xb)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(source->yt, target->yt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(source->yc, target->yc)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_y(source->yb, target->yb)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_z(source->zt, target->zt)))
+	fail++;
+    if (!(N_copy_gradient_neighbours_z(source->zb, target->zb)))
+	fail++;
+
+    if (fail > 0) {
+	return 0;
+    }
+
+    return 1;
+}
+
+/*!
+ * \brief Allocate a N_gradient_field_2d
+ *
+ * The field arrays are of type DCELL. 
+ *
+ * \param rows - number of rows of the 2d array from which the gradient should be calculated
+ * \param cols - number of cols of the 2d array from which the gradient should be calculated
+ * \return N_gradient_field_2d *
+ *
+ * */
+N_gradient_field_2d *N_alloc_gradient_field_2d(int cols, int rows)
+{
+    N_gradient_field_2d *field;
+
+    G_debug(5,
+	    "N_alloc_gradient_field_2d: allocate a N_gradient_field_2d struct");
+
+    field = (N_gradient_field_2d *) G_calloc(1, sizeof(N_gradient_field_2d));
+
+    field->x_array = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    field->y_array = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+
+    field->cols = cols;
+    field->rows = rows;
+
+    return field;
+}
+
+/*!
+ * \brief Free's a N_gradient_neighbours_2d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_field_2d(N_gradient_field_2d * field)
+{
+
+    N_free_array_2d(field->x_array);
+    N_free_array_2d(field->y_array);
+
+    G_free(field);
+
+    field = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Copy N_gradient_field_2d structure from source to target
+ *
+ * \param source - the source N_gradient_field_2d struct
+ * \param target - the target N_gradient_field_2d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_field_2d(N_gradient_field_2d * source,
+			 N_gradient_field_2d * target)
+{
+    G_debug(3, "N_copy_gradient_field_2d: copy N_gradient_field_2d");
+
+    if (!source || !target)
+	return 0;
+
+    N_copy_array_2d(source->x_array, target->x_array);
+    N_copy_array_2d(source->y_array, target->y_array);
+
+    return 1;
+}
+
+/*! \brief Print gradient field information to stdout
+ *
+ * \param field N_gradient_2d_field *
+ * \return void
+ *
+ * */
+void N_print_gradient_field_2d_info(N_gradient_field_2d * field)
+{
+    fprintf(stdout, "N_gradient_field_2d \n");
+    fprintf(stdout, "Cols %i\n", field->cols);
+    fprintf(stdout, "Rows: %i\n", field->rows);
+    fprintf(stdout, "X array pointer: %p\n", field->x_array);
+    fprintf(stdout, "Y array pointer: %p\n", field->y_array);
+    fprintf(stdout, "Min %g\n", field->min);
+    fprintf(stdout, "Max %g\n", field->max);
+    fprintf(stdout, "Sum %g\n", field->sum);
+    fprintf(stdout, "Mean %g\n", field->mean);
+    fprintf(stdout, "Nonull %i\n", field->nonull);
+    fprintf(stdout, "X array info \n");
+    N_print_array_2d_info(field->x_array);
+    fprintf(stdout, "Y array info \n");
+    N_print_array_2d_info(field->y_array);
+
+    return;
+}
+
+
+/*!
+ * \brief Allocate a N_gradient_field_3d
+ *
+ * The field arrays are always of type DCELL_TYPE. 
+ *
+ * \param cols - number of cols of the 3d array from which the gradient should be calculated
+ * \param rows - number of rows of the 3d array from which the gradient should be calculated
+ * \param depths - number of depths of the 3d array from which the gradient should be calculated
+ * \return N_gradient_field_3d *
+ *
+ * */
+N_gradient_field_3d *N_alloc_gradient_field_3d(int cols, int rows, int depths)
+{
+    N_gradient_field_3d *field;
+
+    G_debug(5,
+	    "N_alloc_gradient_field_3d: allocate a N_gradient_field_3d struct");
+
+    field = (N_gradient_field_3d *) G_calloc(1, sizeof(N_gradient_field_3d));
+
+    field->x_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    field->y_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    field->z_array = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+
+    field->cols = cols;
+    field->rows = rows;
+    field->depths = depths;
+
+    return field;
+}
+
+
+/*!
+ * \brief Free's a N_gradient_neighbours_3d structure
+ *
+ * \return void
+ *
+ * */
+void N_free_gradient_field_3d(N_gradient_field_3d * field)
+{
+
+    N_free_array_3d(field->x_array);
+    N_free_array_3d(field->y_array);
+    N_free_array_3d(field->z_array);
+
+    G_free(field);
+
+    field = NULL;
+
+    return;
+}
+
+
+/*!
+ * \brief Copy N_gradient_field_3d structure from source to target
+ *
+ * \param source - the source N_gradient_field_3d struct
+ * \param target - the target N_gradient_field_3d struct
+ * \return int - 1 success, 0 failure while copying
+ *
+ * */
+int
+N_copy_gradient_field_3d(N_gradient_field_3d * source,
+			 N_gradient_field_3d * target)
+{
+    G_debug(3, "N_copy_gradient_field_3d: copy N_gradient_field_3d");
+
+    if (!source || !target)
+	return 0;
+
+    N_copy_array_3d(source->x_array, target->x_array);
+    N_copy_array_3d(source->y_array, target->y_array);
+    N_copy_array_3d(source->z_array, target->z_array);
+
+    return 1;
+}
+
+/*! \brief Print gradient field information to stdout
+ *
+ * \param field N_gradient_3d_field *
+ * \return void
+ *
+ * */
+void N_print_gradient_field_3d_info(N_gradient_field_3d * field)
+{
+
+    fprintf(stdout, "N_gradient_field_3d \n");
+    fprintf(stdout, "Cols %i\n", field->cols);
+    fprintf(stdout, "Rows: %i\n", field->rows);
+    fprintf(stdout, "Depths %i\n", field->depths);
+    fprintf(stdout, "X array pointer: %p\n", field->x_array);
+    fprintf(stdout, "Y array pointer: %p\n", field->y_array);
+    fprintf(stdout, "Z array pointer: %p\n", field->z_array);
+    fprintf(stdout, "Min %g\n", field->min);
+    fprintf(stdout, "Max %g\n", field->max);
+    fprintf(stdout, "Sum %g\n", field->sum);
+    fprintf(stdout, "Mean %g\n", field->mean);
+    fprintf(stdout, "Nonull %i\n", field->nonull);
+    fprintf(stdout, "X array info \n");
+    N_print_array_3d_info(field->x_array);
+    fprintf(stdout, "Y array info \n");
+    N_print_array_3d_info(field->y_array);
+    fprintf(stdout, "Z array info \n");
+    N_print_array_3d_info(field->z_array);
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_gradient_calc.c (from rev 62429, grass/trunk/lib/gpde/N_gradient_calc.c)
===================================================================
--- grass/trunk/lib/gpde/n_gradient_calc.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_gradient_calc.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,657 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:     	gradient management functions 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <grass/N_pde.h>
+
+/*! \brief Calculate basic statistics of a gradient field
+ *
+ * The statistic is stored in the gradient field struct
+ *
+ * \param field N_gradient_2d_field *
+ * \return void
+ *
+ * */
+void N_calc_gradient_field_2d_stats(N_gradient_field_2d * field)
+{
+    double minx, miny;
+    double maxx, maxy;
+    double sumx, sumy;
+    int nonullx, nonully;
+
+    G_debug(3,
+	    "N_calc_gradient_field_2d_stats: compute gradient field stats");
+
+    N_calc_array_2d_stats(field->x_array, &minx, &maxx, &sumx, &nonullx, 0);
+    N_calc_array_2d_stats(field->y_array, &miny, &maxy, &sumy, &nonully, 0);
+
+    if (minx < miny)
+	field->min = minx;
+    else
+	field->min = miny;
+
+    if (maxx > maxy)
+	field->max = maxx;
+    else
+	field->max = maxy;
+
+    field->sum = sumx + sumy;
+    field->nonull = nonullx + nonully;
+    field->mean = field->sum / (double)field->nonull;
+
+    return;
+}
+
+/*!
+ * \brief This function computes the gradient based on the input N_array_2d pot
+ * (potential), a weighting factor N_array_2d named weight and the distance between two cells 
+ * saved in the N_geom_data struct.
+ *
+ * The gradient is calculated between cells for each cell and direction.
+ * An existing gradient field can be filled with new data or, if a NULL pointer is
+ * given, a new gradient field will be allocated with the appropriate size.
+ *
+ *
+ \verbatim
+ ______________ 
+ |    |    |    |
+ |    |    |    |
+ |----|-NC-|----|
+ |    |    |    |
+ |   WC    EC   |
+ |    |    |    |
+ |----|-SC-|----|
+ |    |    |    |
+ |____|____|____|
+
+
+ x - direction:
+
+ r = 2 * weight[row][col]*weight[row][col + 1] / (weight[row][col]*weight[row][col + 1])
+ EC = r * (pot[row][col] - pot[row][col + 1])/dx
+
+ y - direction:
+
+ r = 2 * weight[row][col]*weight[row + 1][col] / (weight[row][col]*weight[row + 1][col])
+ SC = r * (pot[row][col] - pot[row + 1][col])/dy
+
+ the values SC and EC are the values of the next row/col
+
+
+ \endverbatim
+ * \param pot N_array_2d * - the potential N_array_2d 
+ * \param weight_x N_array_2d * - the weighting factor N_array_2d used to modify the gradient in x-direction
+ * \param weight_y N_array_2d * - the weighting factor N_array_2d used to modify the gradient in y-direction
+ * \param geom N_geom_data * - geometry data structure
+ * \param gradfield N_gradient_field_2d * - a gradient field of the correct size, if a NULL pointer is provided this gradient field will be new allocated
+ * \return N_gradient_field_2d * - the pointer to the computed gradient field
+
+ *
+ * */
+N_gradient_field_2d *N_compute_gradient_field_2d(N_array_2d * pot,
+						 N_array_2d * weight_x,
+						 N_array_2d * weight_y,
+						 N_geom_data * geom,
+						 N_gradient_field_2d *
+						 gradfield)
+{
+    int i, j;
+    int rows, cols;
+    double dx, dy, p1, p2, r1, r2, mean, grad, res;
+    N_gradient_field_2d *field = gradfield;
+
+
+    if (pot->cols != weight_x->cols || pot->cols != weight_y->cols)
+	G_fatal_error
+	    ("N_compute_gradient_field_2d: the arrays are not of equal size");
+
+    if (pot->rows != weight_x->rows || pot->rows != weight_y->rows)
+	G_fatal_error
+	    ("N_compute_gradient_field_2d: the arrays are not of equal size");
+
+    if (pot->cols != geom->cols || pot->rows != geom->rows)
+	G_fatal_error
+	    ("N_compute_gradient_field_2d: array sizes and geometry data are different");
+
+
+    G_debug(3, "N_compute_gradient_field_2d: compute gradient field");
+
+    rows = pot->rows;
+    cols = pot->cols;
+    dx = geom->dx;
+    dy = geom->dy;
+
+    if (field == NULL) {
+	field = N_alloc_gradient_field_2d(cols, rows);
+    }
+    else {
+	if (field->cols != geom->cols || field->rows != geom->rows)
+	    G_fatal_error
+		("N_compute_gradient_field_2d: gradient field sizes and geometry data are different");
+    }
+
+
+    for (j = 0; j < rows; j++)
+	for (i = 0; i < cols - 1; i++) {
+	    grad = 0;
+	    mean = 0;
+
+	    /* Only compute if the arrays are not null */
+	    if (!N_is_array_2d_value_null(pot, i, j) &&
+		!N_is_array_2d_value_null(pot, i + 1, j)) {
+		p1 = N_get_array_2d_d_value(pot, i, j);
+		p2 = N_get_array_2d_d_value(pot, i + 1, j);
+		grad = (p1 - p2) / dx;	/* gradient */
+	    }
+	    if (!N_is_array_2d_value_null(weight_x, i, j) &&
+		!N_is_array_2d_value_null(weight_x, i + 1, j)) {
+		r1 = N_get_array_2d_d_value(weight_x, i, j);
+		r2 = N_get_array_2d_d_value(weight_x, i + 1, j);
+		mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
+	    }
+
+	    res = mean * grad;
+
+	    N_put_array_2d_d_value(field->x_array, i + 1, j, res);
+
+	}
+
+    for (j = 0; j < rows - 1; j++)
+	for (i = 0; i < cols; i++) {
+	    grad = 0;
+	    mean = 0;
+
+	    /* Only compute if the arrays are not null */
+	    if (!N_is_array_2d_value_null(pot, i, j) &&
+		!N_is_array_2d_value_null(pot, i, j + 1)) {
+		p1 = N_get_array_2d_d_value(pot, i, j);
+		p2 = N_get_array_2d_d_value(pot, i, j + 1);
+		grad = (p1 - p2) / dy;	/* gradient */
+	    }
+	    if (!N_is_array_2d_value_null(weight_y, i, j) &&
+		!N_is_array_2d_value_null(weight_y, i, j + 1)) {
+		r1 = N_get_array_2d_d_value(weight_y, i, j);
+		r2 = N_get_array_2d_d_value(weight_y, i, j + 1);
+		mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
+	    }
+
+	    res = -1 * mean * grad;
+
+	    N_put_array_2d_d_value(field->y_array, i, j + 1, res);
+
+	}
+
+    /*Compute gradient field statistics */
+    N_calc_gradient_field_2d_stats(field);
+
+    return field;
+}
+
+/*! 
+ * \brief Calculate the x and y vector components from a gradient field for each 
+ * cell and stores them in the provided N_array_2d structures
+ *
+ * The arrays must have the same size as the gradient field.
+
+ \verbatim
+
+ Based on this storages scheme the gradient vector for each cell is 
+ calculated and stored in the provided  N_array_2d structures
+
+ ______________ 
+ |    |    |    |
+ |    |    |    |
+ |----|-NC-|----|
+ |    |    |    |
+ |   WC    EC   |
+ |    |    |    |
+ |----|-SC-|----|
+ |    |    |    |
+ |____|____|____|
+
+ x vector component:
+
+ x = (WC + EC) / 2
+
+ y vector component:
+
+ y = (NC + SC) / 2
+
+ \endverbatim
+ *
+ * \param field N_gradient_field_2d *
+ * \param x_comp N_array_2d * - the array in which the x component will be written
+ * \param y_comp N_array_2d * - the array in which the y component will be written
+ *
+ * \return void
+ * */
+void
+N_compute_gradient_field_components_2d(N_gradient_field_2d * field,
+				       N_array_2d * x_comp,
+				       N_array_2d * y_comp)
+{
+    int i, j;
+
+    int rows, cols;
+
+    double vx, vy;
+
+    N_array_2d *x = x_comp;
+
+    N_array_2d *y = y_comp;
+
+    N_gradient_2d grad;
+
+
+    if (!x)
+	G_fatal_error("N_compute_gradient_components_2d: x array is empty");
+    if (!y)
+	G_fatal_error("N_compute_gradient_components_2d: y array is empty");
+
+    cols = field->x_array->cols;
+    rows = field->x_array->rows;
+
+    /*Check the array sizes */
+    if (x->cols != cols || x->rows != rows)
+	G_fatal_error
+	    ("N_compute_gradient_components_2d: the size of the x array doesn't fit the gradient field size");
+    if (y->cols != cols || y->rows != rows)
+	G_fatal_error
+	    ("N_compute_gradient_components_2d: the size of the y array doesn't fit the gradient field size");
+
+    for (j = 0; j < rows; j++)
+	for (i = 0; i < cols; i++) {
+	    N_get_gradient_2d(field, &grad, i, j);
+
+	    /* in case a gradient is zero, we expect a no flow boundary */
+	    if (grad.WC == 0.0 || grad.EC == 0.0)
+		vx = (grad.WC + grad.EC);
+	    else
+		vx = (grad.WC + grad.EC) / 2;
+	    if (grad.NC == 0.0 || grad.SC == 0.0)
+		vy = (grad.NC + grad.SC);
+	    else
+		vy = (grad.NC + grad.SC) / 2;
+
+	    N_put_array_2d_d_value(x, i, j, vx);
+	    N_put_array_2d_d_value(y, i, j, vy);
+	}
+
+    return;
+}
+
+/*! \brief Calculate basic statistics of a gradient field
+ *
+ * The statistic is stored in the gradient field struct
+ *
+ * \param field N_gradient_3d_field *
+ * \return void
+ *
+ * */
+void N_calc_gradient_field_3d_stats(N_gradient_field_3d * field)
+{
+    double minx, miny, minz;
+
+    double maxx, maxy, maxz;
+
+    double sumx, sumy, sumz;
+
+    int nonullx, nonully, nonullz;
+
+    G_debug(3,
+	    "N_calc_gradient_field_3d_stats: compute gradient field stats");
+
+    N_calc_array_3d_stats(field->x_array, &minx, &maxx, &sumx, &nonullx, 0);
+    N_calc_array_3d_stats(field->y_array, &miny, &maxy, &sumy, &nonully, 0);
+    N_calc_array_3d_stats(field->z_array, &minz, &maxz, &sumz, &nonullz, 0);
+
+    if (minx <= minz && minx <= miny)
+	field->min = minx;
+    if (miny <= minz && miny <= minx)
+	field->min = miny;
+    if (minz <= minx && minz <= miny)
+	field->min = minz;
+
+    if (maxx >= maxz && maxx >= maxy)
+	field->max = maxx;
+    if (maxy >= maxz && maxy >= maxx)
+	field->max = maxy;
+    if (maxz >= maxx && maxz >= maxy)
+	field->max = maxz;
+
+    field->sum = sumx + sumy + sumz;
+    field->nonull = nonullx + nonully + nonullz;
+    field->mean = field->sum / (double)field->nonull;
+
+    return;
+}
+
+
+/*!
+ * \brief This function computes the gradient based on the input N_array_3d pot
+ * (that means potential), a weighting factor N_array_3d named weight and the distance between two cells 
+ * saved in the N_geom_data struct.
+ *
+ * The gradient is calculated between cells for each cell and direction.
+ * An existing gradient field can be filled with new data or, if a NULL pointer is
+ * given, a new gradient field will be allocated with the appropriate size.
+ *
+ *
+ *
+ *
+ \verbatim
+
+ |  /
+ TC NC
+ |/
+ --WC-----EC--
+ /|
+ SC BC
+ /  |
+
+ x - direction:
+
+ r = 2 * weight_x[depth][row][col]*weight_x[depth][row][col + 1] / (weight_X[depth][row][col]*weight_x[depth][row][col + 1])
+ EC = r * (pot[depth][row][col] - pot[depth][row][col + 1])/dx
+
+ y - direction:
+
+ r = 2 * weight_y[depth][row][col]*weight_y[depth][row + 1][col] / (weight_y[depth][row][col]*weight_y[depth][row + 1][col])
+ SC = r * (pot[depth][row][col] - pot[depth][row + 1][col])/dy
+
+ z - direction:
+
+ r = 2 * weight_z[depth][row][col]*weight_z[depth + 1][row][col] / (weight_z[depth][row][col]*weight_z[depth + 1][row][col])
+ TC = r * (pot[depth][row][col] - pot[depth + 1][row][col])/dy
+
+ the values BC, NC, WC are the values of the next depth/row/col
+
+
+ \endverbatim
+ * \param pot N_array_3d * - the potential N_array_2d 
+ * \param weight_x N_array_3d * - the weighting factor N_array_3d used to modify the gradient in x-direction
+ * \param weight_y N_array_3d * - the weighting factor N_array_3d used to modify the gradient in y-direction
+ * \param weight_z N_array_3d * - the weighting factor N_array_3d used to modify the gradient in z-direction
+ * \param geom N_geom_data * - geometry data structure
+ * \param gradfield N_gradient_field_3d * - a gradient field of the correct size, if a NULL pointer is provided this gradient field will be new allocated
+ * \return N_gradient_field_3d * - the pointer to the computed gradient field
+ *
+ * */
+N_gradient_field_3d *N_compute_gradient_field_3d(N_array_3d * pot,
+						 N_array_3d * weight_x,
+						 N_array_3d * weight_y,
+						 N_array_3d * weight_z,
+						 N_geom_data * geom,
+						 N_gradient_field_3d *
+						 gradfield)
+{
+    int i, j, k;
+
+    int cols, rows, depths;
+
+    double dx, dy, dz, p1, p2, r1, r2, mean, grad, res;
+
+    N_gradient_field_3d *field = gradfield;
+
+
+    if (pot->cols != weight_x->cols || pot->cols != weight_y->cols ||
+	pot->cols != weight_z->cols)
+	G_fatal_error
+	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
+
+    if (pot->rows != weight_x->rows || pot->rows != weight_y->rows ||
+	pot->rows != weight_z->rows)
+	G_fatal_error
+	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
+
+    if (pot->depths != weight_x->depths || pot->depths != weight_y->depths ||
+	pot->depths != weight_z->depths)
+	G_fatal_error
+	    ("N_compute_gradient_field_3d: the arrays are not of equal size");
+
+    if (pot->cols != geom->cols || pot->rows != geom->rows ||
+	pot->depths != geom->depths)
+	G_fatal_error
+	    ("N_compute_gradient_field_3d: array sizes and geometry data are different");
+
+    G_debug(3, "N_compute_gradient_field_3d: compute gradient field");
+
+    cols = geom->cols;
+    rows = geom->rows;
+    depths = geom->depths;
+    dx = geom->dx;
+    dy = geom->dy;
+    dz = geom->dz;
+
+    if (gradfield == NULL) {
+	field = N_alloc_gradient_field_3d(cols, rows, depths);
+    }
+    else {
+	if (field->cols != geom->cols || field->rows != geom->rows ||
+	    field->depths != geom->depths)
+	    G_fatal_error
+		("N_compute_gradient_field_3d: gradient field sizes and geometry data are different");
+    }
+
+    for (k = 0; k < depths; k++)
+	for (j = 0; j < rows; j++)
+	    for (i = 0; i < cols - 1; i++) {
+		grad = 0;
+		mean = 0;
+
+		/*Only compute if the arrays are not null */
+		if (!N_is_array_3d_value_null(pot, i, j, k) &&
+		    !N_is_array_3d_value_null(pot, i + 1, j, k)) {
+		    p1 = N_get_array_3d_d_value(pot, i, j, k);
+		    p2 = N_get_array_3d_d_value(pot, i + 1, j, k);
+		    grad = (p1 - p2) / dx;	/* gradient */
+		}
+		if (!N_is_array_3d_value_null(weight_x, i, j, k) &&
+		    !N_is_array_3d_value_null(weight_x, i + 1, j, k)) {
+		    r1 = N_get_array_3d_d_value(weight_x, i, j, k);
+		    r2 = N_get_array_3d_d_value(weight_x, i + 1, j, k);
+		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
+		}
+
+		res = mean * grad;
+
+		G_debug(6,
+			"N_compute_gradient_field_3d: X-direction insert value %6.5g at %i %i %i ",
+			res, k, j, i + 1);
+
+		N_put_array_3d_d_value(field->x_array, i + 1, j, k, res);
+
+	    }
+
+    for (k = 0; k < depths; k++)
+	for (j = 0; j < rows - 1; j++)
+	    for (i = 0; i < cols; i++) {
+		grad = 0;
+		mean = 0;
+
+		/* Only compute if the arrays are not null */
+		if (!N_is_array_3d_value_null(pot, i, j, k) &&
+		    !N_is_array_3d_value_null(pot, i, j + 1, k)) {
+		    p1 = N_get_array_3d_d_value(pot, i, j, k);
+		    p2 = N_get_array_3d_d_value(pot, i, j + 1, k);
+		    grad = (p1 - p2) / dy;	/* gradient */
+		}
+		if (!N_is_array_3d_value_null(weight_y, i, j, k) &&
+		    !N_is_array_3d_value_null(weight_y, i, j + 1, k)) {
+		    r1 = N_get_array_3d_d_value(weight_y, i, j, k);
+		    r2 = N_get_array_3d_d_value(weight_y, i, j + 1, k);
+		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
+		}
+
+		res = -1 * mean * grad;	/*invert the direction, because we count from north to south,
+					 * but the gradient is defined in y direction */
+
+		G_debug(6,
+			"N_compute_gradient_field_3d: Y-direction insert value %6.5g at %i %i %i ",
+			res, k, j + 1, i);
+
+		N_put_array_3d_d_value(field->y_array, i, j + 1, k, res);
+
+	    }
+
+    for (k = 0; k < depths - 1; k++)
+	for (j = 0; j < rows; j++)
+	    for (i = 0; i < cols; i++) {
+		grad = 0;
+		mean = 0;
+
+		/* Only compute if the arrays are not null */
+		if (!N_is_array_3d_value_null(pot, i, j, k) &&
+		    !N_is_array_3d_value_null(pot, i, j, k + 1)) {
+		    p1 = N_get_array_3d_d_value(pot, i, j, k);
+		    p2 = N_get_array_3d_d_value(pot, i, j, k + 1);
+		    grad = (p1 - p2) / dz;	/* gradient */
+		}
+		if (!N_is_array_3d_value_null(weight_z, i, j, k) &&
+		    !N_is_array_3d_value_null(weight_z, i, j, k + 1)) {
+		    r1 = N_get_array_3d_d_value(weight_z, i, j, k);
+		    r2 = N_get_array_3d_d_value(weight_z, i, j, k + 1);
+		    mean = N_calc_harmonic_mean(r1, r2);	/*harmonical mean */
+		}
+
+		res = mean * grad;
+
+		G_debug(6,
+			"N_compute_gradient_field_3d: Z-direction insert value %6.5g at %i %i %i ",
+			res, k + 1, j, i);
+
+		N_put_array_3d_d_value(field->z_array, i, j, k + 1, res);
+
+	    }
+
+    /*Compute gradient field statistics */
+    N_calc_gradient_field_3d_stats(field);
+
+    return field;
+}
+
+/*! 
+ * \brief Calculate the x, y and z vector components from a gradient field for each cell 
+ *  and store them in the provided N_array_3d structures
+ *
+ * The arrays must have the same size as the gradient field.
+ *
+ \verbatim
+
+ Based on this storages scheme the gradient vector for each cell is 
+ calculated and stored in the provided  N_array_3d structures
+
+
+ |  /
+ TC NC
+ |/
+ --WC-----EC--
+ /|
+ SC BC
+ /  |
+
+
+ x vector component:
+
+ x = (WC + EC) / 2
+
+ y vector component:
+
+ y = (NC + SC) / 2
+
+ z vector component:
+
+ z = (TC + BC) / 2
+
+ \endverbatim
+
+ * \param field N_gradient_field_3d *
+ * \param x_comp N_array_3d * - the array in which the x component will be written
+ * \param y_comp N_array_3d * - the array in which the y component will be written
+ * \param z_comp N_array_3d * - the array in which the z component will be written
+ *
+ * \return void
+ * */
+void
+N_compute_gradient_field_components_3d(N_gradient_field_3d * field,
+				       N_array_3d * x_comp,
+				       N_array_3d * y_comp,
+				       N_array_3d * z_comp)
+{
+    int i, j, k;
+
+    int rows, cols, depths;
+
+    double vx, vy, vz;
+
+    N_array_3d *x = x_comp;
+
+    N_array_3d *y = y_comp;
+
+    N_array_3d *z = z_comp;
+
+    N_gradient_3d grad;
+
+
+    if (!x)
+	G_fatal_error("N_compute_gradient_components_3d: x array is empty");
+    if (!y)
+	G_fatal_error("N_compute_gradient_components_3d: y array is empty");
+    if (!z)
+	G_fatal_error("N_compute_gradient_components_3d: z array is empty");
+
+    cols = field->x_array->cols;
+    rows = field->x_array->rows;
+    depths = field->x_array->depths;
+
+    /*Check the array sizes */
+    if (x->cols != cols || x->rows != rows || x->depths != depths)
+	G_fatal_error
+	    ("N_compute_gradient_components_3d: the size of the x array doesn't fit the gradient field size");
+    if (y->cols != cols || y->rows != rows || y->depths != depths)
+	G_fatal_error
+	    ("N_compute_gradient_components_3d: the size of the y array doesn't fit the gradient field size");
+    if (z->cols != cols || z->rows != rows || z->depths != depths)
+	G_fatal_error
+	    ("N_compute_gradient_components_3d: the size of the z array doesn't fit the gradient field size");
+
+    for (k = 0; k < depths; k++)
+	for (j = 0; j < rows; j++)
+	    for (i = 0; i < cols; i++) {
+		N_get_gradient_3d(field, &grad, i, j, k);
+		/* in case a gradient is zero, we expect a no flow boundary */
+		if (grad.WC == 0.0 || grad.EC == 0.0)
+		    vx = (grad.WC + grad.EC);
+		else
+		    vx = (grad.WC + grad.EC) / 2;
+		if (grad.NC == 0.0 || grad.SC == 0.0)
+		    vy = (grad.NC + grad.SC);
+		else
+		    vy = (grad.NC + grad.SC) / 2;
+		if (grad.TC == 0.0 || grad.BC == 0.0)
+		    vz = (grad.TC + grad.BC);
+		else
+		    vz = (grad.TC + grad.BC) / 2;
+
+		N_put_array_3d_d_value(x, i, j, k, vx);
+		N_put_array_3d_d_value(y, i, j, k, vy);
+		N_put_array_3d_d_value(z, i, j, k, vz);
+	    }
+
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_gwflow.c (from rev 62429, grass/trunk/lib/gpde/N_gwflow.c)
===================================================================
--- grass/trunk/lib/gpde/n_gwflow.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_gwflow.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,720 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      groundwater flow in porous media 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <grass/N_gwflow.h>
+
+/* *************************************************************** */
+/* ***************** N_gwflow_data3d ***************************** */
+/* *************************************************************** */
+/*!
+ * \brief Alllocate memory for the groundwater calculation data structure in 3 dimensions
+ *
+ * The groundwater calculation data structure will be allocated including
+ * all appendant 3d and 2d arrays. The offset for the 3d arrays is one
+ * to establish homogeneous Neumann boundary conditions at the calculation area border.
+ * This data structure is used to create a linear equation system based on the computation of
+ * groundwater flow in porous media with the finite volume method.
+ *
+ * \param cols   int
+ * \param rows   int
+ * \param depths int
+ * \return N_gwflow_data3d *
+ * */
+N_gwflow_data3d *N_alloc_gwflow_data3d(int cols, int rows, int depths,
+				       int river, int drain)
+{
+    N_gwflow_data3d *data;
+
+    data = (N_gwflow_data3d *) G_calloc(1, sizeof(N_gwflow_data3d));
+
+    data->phead = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->phead_start = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->status = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->hc_x = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->hc_y = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->hc_z = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->q = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->s = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->nf = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->r = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+
+    if (river) {
+	data->river_head =
+	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+	data->river_leak =
+	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+	data->river_bed = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    }
+    else {
+	data->river_head = NULL;
+	data->river_leak = NULL;
+	data->river_bed = NULL;
+    }
+
+    if (drain) {
+	data->drain_leak =
+	    N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+	data->drain_bed = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    }
+    else {
+	data->drain_leak = NULL;
+	data->drain_bed = NULL;
+    }
+
+    return data;
+}
+
+/* *************************************************************** */
+/* ********************* N_free_gwflow_data3d ******************** */
+/* *************************************************************** */
+/*!
+ * \brief Release the memory of the groundwater flow data structure in three dimensions
+ *
+ * \param data N_gwflow_data3d *
+ * \return void *
+ * */
+
+void N_free_gwflow_data3d(N_gwflow_data3d * data)
+{
+    if (data->phead)
+	N_free_array_3d(data->phead);
+    if (data->phead_start)
+	N_free_array_3d(data->phead_start);
+    if (data->status)
+	N_free_array_3d(data->status);
+    if (data->hc_x)
+	N_free_array_3d(data->hc_x);
+    if (data->hc_y)
+	N_free_array_3d(data->hc_y);
+    if (data->hc_z)
+	N_free_array_3d(data->hc_z);
+    if (data->q)
+	N_free_array_3d(data->q);
+    if (data->s)
+	N_free_array_3d(data->s);
+    if (data->nf)
+	N_free_array_3d(data->nf);
+    if (data->r)
+	N_free_array_2d(data->r);
+    if (data->river_head)
+	N_free_array_3d(data->river_head);
+    if (data->river_leak)
+	N_free_array_3d(data->river_leak);
+    if (data->river_bed)
+	N_free_array_3d(data->river_bed);
+    if (data->drain_leak)
+	N_free_array_3d(data->drain_leak);
+    if (data->drain_bed)
+	N_free_array_3d(data->drain_bed);
+
+    G_free(data);
+
+    data = NULL;
+
+    return;
+}
+
+/* *************************************************************** */
+/* ******************** N_alloc_gwflow_data2d ******************** */
+/* *************************************************************** */
+/*!
+ * \brief Alllocate memory for the groundwater calculation data structure in 2 dimensions
+ * 
+ * The groundwater calculation data structure will be allocated including
+ * all appendant 2d arrays. The offset for the 3d arrays is one
+ * to establish homogeneous Neumann boundary conditions at the calculation area border.
+ * This data structure is used to create a linear equation system based on the computation of
+ * groundwater flow in porous media with the finite volume method.
+ *
+ * \param cols int
+ * \param rows int
+ * \return N_gwflow_data2d *
+ * */
+N_gwflow_data2d *N_alloc_gwflow_data2d(int cols, int rows, int river,
+				       int drain)
+{
+    N_gwflow_data2d *data;
+
+    data = (N_gwflow_data2d *) G_calloc(1, sizeof(N_gwflow_data2d));
+
+    data->phead = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->phead_start = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->status = N_alloc_array_2d(cols, rows, 1, CELL_TYPE);
+    data->hc_x = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->hc_y = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->q = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->s = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->nf = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->r = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->top = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->bottom = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+
+    if (river) {
+	data->river_head = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+	data->river_leak = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+	data->river_bed = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    }
+    else {
+	data->river_head = NULL;
+	data->river_leak = NULL;
+	data->river_bed = NULL;
+    }
+
+    if (drain) {
+	data->drain_leak = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+	data->drain_bed = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    }
+    else {
+	data->drain_leak = NULL;
+	data->drain_bed = NULL;
+    }
+
+
+    return data;
+}
+
+/* *************************************************************** */
+/* ****************** N_free_gwflow_data2d *********************** */
+/* *************************************************************** */
+/*!
+ * \brief Release the memory of the groundwater flow data structure in two dimensions
+ *
+ * \param data N_gwflow_data2d *
+ * \return void
+ * */
+void N_free_gwflow_data2d(N_gwflow_data2d * data)
+{
+    if (data->phead)
+	N_free_array_2d(data->phead);
+    if (data->phead_start)
+	N_free_array_2d(data->phead_start);
+    if (data->status)
+	N_free_array_2d(data->status);
+    if (data->hc_x)
+	N_free_array_2d(data->hc_x);
+    if (data->hc_y)
+	N_free_array_2d(data->hc_y);
+    if (data->q)
+	N_free_array_2d(data->q);
+    if (data->s)
+	N_free_array_2d(data->s);
+    if (data->nf)
+	N_free_array_2d(data->nf);
+    if (data->r)
+	N_free_array_2d(data->r);
+    if (data->top)
+	N_free_array_2d(data->top);
+    if (data->bottom)
+	N_free_array_2d(data->bottom);
+    if (data->river_head)
+	N_free_array_2d(data->river_head);
+    if (data->river_leak)
+	N_free_array_2d(data->river_leak);
+    if (data->river_bed)
+	N_free_array_2d(data->river_bed);
+    if (data->drain_leak)
+	N_free_array_2d(data->drain_leak);
+    if (data->drain_bed)
+	N_free_array_2d(data->drain_bed);
+
+    G_free(data);
+
+    data = NULL;;
+
+    return;
+}
+
+/* *************************************************************** */
+/* ***************** N_callback_gwflow_3d ************************ */
+/* *************************************************************** */
+/*!
+ * \brief This callback function creates the mass balance of a 7 point star
+ *
+ * The mass balance is based on the common groundwater flow equation:
+ *
+ * \f[Ss \frac{\partial h}{\partial t} = \nabla {\bf K} \nabla h + q \f]
+ *
+ * This equation is discretizised with the finite volume method in three dimensions.
+ *
+ *
+ * \param gwdata N_gwflow_data3d *
+ * \param geom N_geom_data *
+ * \param col   int
+ * \param row   int
+ * \param depth int
+ * \return N_data_star *
+ *
+ * */
+N_data_star *N_callback_gwflow_3d(void *gwdata, N_geom_data * geom, int col,
+				  int row, int depth)
+{
+    double hc_e = 0, hc_w = 0, hc_n = 0, hc_s = 0, hc_t = 0, hc_b = 0;
+    double dx, dy, dz, Ax, Ay, Az;
+    double hc_x, hc_y, hc_z;
+    double hc_xw, hc_yn, hc_zt;
+    double hc_xe, hc_ys, hc_zb;
+    double hc_start;
+    double Ss, r, nf, q;
+    double C, W, E, N, S, T, B, V;
+    N_data_star *mat_pos;
+    N_gwflow_data3d *data;
+
+    /*cast the void pointer to the right data structure */
+    data = (N_gwflow_data3d *) gwdata;
+
+    dx = geom->dx;
+    dy = geom->dy;
+    dz = geom->dz;
+    Az = N_get_geom_data_area_of_cell(geom, row);
+    Ay = geom->dx * geom->dz;
+    Ax = geom->dz * geom->dy;
+
+    /*read the data from the arrays */
+    hc_start = N_get_array_3d_d_value(data->phead_start, col, row, depth);
+
+    hc_x = N_get_array_3d_d_value(data->hc_x, col, row, depth);
+    hc_y = N_get_array_3d_d_value(data->hc_y, col, row, depth);
+    hc_z = N_get_array_3d_d_value(data->hc_z, col, row, depth);
+
+    hc_xw = N_get_array_3d_d_value(data->hc_x, col - 1, row, depth);
+    hc_xe = N_get_array_3d_d_value(data->hc_x, col + 1, row, depth);
+    hc_yn = N_get_array_3d_d_value(data->hc_y, col, row - 1, depth);
+    hc_ys = N_get_array_3d_d_value(data->hc_y, col, row + 1, depth);
+    hc_zt = N_get_array_3d_d_value(data->hc_z, col, row, depth + 1);
+    hc_zb = N_get_array_3d_d_value(data->hc_z, col, row, depth - 1);
+
+    hc_w = N_calc_harmonic_mean(hc_xw, hc_x);
+    hc_e = N_calc_harmonic_mean(hc_xe, hc_x);
+    hc_n = N_calc_harmonic_mean(hc_yn, hc_y);
+    hc_s = N_calc_harmonic_mean(hc_ys, hc_y);
+    hc_t = N_calc_harmonic_mean(hc_zt, hc_z);
+    hc_b = N_calc_harmonic_mean(hc_zb, hc_z);
+
+    /*inner sources */
+    q = N_get_array_3d_d_value(data->q, col, row, depth);
+    /*storativity */
+    Ss = N_get_array_3d_d_value(data->s, col, row, depth);
+    /*porosity */
+    nf = N_get_array_3d_d_value(data->nf, col, row, depth);
+
+    /*mass balance center cell to western cell */
+    W = -1 * Ax * hc_w / dx;
+    /*mass balance center cell to eastern cell */
+    E = -1 * Ax * hc_e / dx;
+    /*mass balance center cell to northern cell */
+    N = -1 * Ay * hc_n / dy;
+    /*mass balance center cell to southern cell */
+    S = -1 * Ay * hc_s / dy;
+    /*mass balance center cell to top cell */
+    T = -1 * Az * hc_t / dz;
+    /*mass balance center cell to bottom cell */
+    B = -1 * Az * hc_b / dz;
+
+    /*storativity */
+    Ss = Az * dz * Ss;
+
+    /*the diagonal entry of the matrix */
+    C = -1 * (W + E + N + S + T + B - Ss / data->dt * Az);
+
+    /*the entry in the right side b of Ax = b */
+    V = (q + hc_start * Ss / data->dt * Az);
+
+    /*only the top cells will have recharge */
+    if (depth == geom->depths - 2) {
+	r = N_get_array_2d_d_value(data->r, col, row);
+	V += r * Az;
+    }
+
+    G_debug(5, "N_callback_gwflow_3d: called [%i][%i][%i]", depth, col, row);
+
+    /*create the 7 point star entries */
+    mat_pos = N_create_7star(C, W, E, N, S, T, B, V);
+
+    return mat_pos;
+}
+
+
+/* *************************************************************** */
+/* ****************** N_gwflow_3d_calc_water_budget ************** */
+/* *************************************************************** */
+/*!
+ * \brief This function computes the water budget of the entire groundwater
+ *
+ * The water budget is calculated for each active and dirichlet cell from
+ * its surrounding neighbours. This is based on the 7 star mass balance computation
+ * of N_callback_gwflow_3d and the gradient of the water heights in the cells.
+ * The sum of the water budget of each active/dirichlet cell must be near zero
+ * due the effect of numerical inaccuracy of cpu's.
+ *
+ * \param gwdata N_gwflow_data3d *
+ * \param geom N_geom_data *
+ * \param budget N_array_3d
+ * \return void
+ *
+ * */
+void
+N_gwflow_3d_calc_water_budget(N_gwflow_data3d * data, N_geom_data * geom, N_array_3d * budget)
+{
+    int z, y, x, stat;
+    double h, hc;
+    double val;
+    double sum;
+    N_data_star *dstar;
+
+    int rows = data->status->rows;
+    int cols = data->status->cols;
+    int depths = data->status->depths;
+    sum = 0;
+
+    for (z = 0; z < depths; z++) {
+        for (y = 0; y < rows; y++) {
+            G_percent(y, rows - 1, 10);
+            for (x = 0; x < cols; x++) {
+                stat = (int)N_get_array_3d_d_value(data->status, x, y, z);
+
+                val = 0.0;
+
+                if (stat != N_CELL_INACTIVE ) {	/*all active/dirichlet cells */
+
+                    /* Compute the flow parameter */
+                    dstar = N_callback_gwflow_3d(data, geom, x, y, z);
+                    /* Compute the gradient in each direction pointing from the center */
+                    hc = N_get_array_3d_d_value(data->phead, x, y, z);
+
+                    if((int)N_get_array_3d_d_value(data->status, x + 1, y    , z) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x + 1, y    , z);
+                        val += dstar->E * (hc - h);
+                    }
+                    if((int)N_get_array_3d_d_value(data->status, x - 1, y    , z) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x - 1, y    , z);
+                        val += dstar->W * (hc - h);
+                    }
+                    if((int)N_get_array_3d_d_value(data->status, x    , y + 1, z) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x    , y + 1, z);
+                        val += dstar->S * (hc - h);
+                    }
+                    if((int)N_get_array_3d_d_value(data->status, x    , y - 1, z) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x    , y - 1, z);
+                        val += dstar->N * (hc - h);
+                    }
+                    if((int)N_get_array_3d_d_value(data->status, x    , y    , z + 1) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x    , y    , z + 1);
+                        val += dstar->T * (hc - h);
+                    }
+                    if((int)N_get_array_3d_d_value(data->status, x    , y    , z - 1) != N_CELL_INACTIVE) {
+                        h = N_get_array_3d_d_value(data->phead,  x    , y    , z - 1);
+                        val += dstar->B * (hc - h);
+                    }
+                    sum += val;
+
+                    G_free(dstar);
+                }
+                else {
+                    Rast_set_null_value(&val, 1, DCELL_TYPE);
+                }
+                N_put_array_3d_d_value(budget, x, y, z, val);
+            }
+        }
+    }
+
+    if(fabs(sum) < 0.0000000001)
+        G_message(_("The total sum of the water budget: %g\n"), sum);
+    else
+        G_warning(_("The total sum of the water budget is significantly larger then 0: %g\n"), sum);
+
+    return;
+}
+
+
+
+/* *************************************************************** */
+/* ****************** N_callback_gwflow_2d *********************** */
+/* *************************************************************** */
+/*!
+ * \brief This callback function creates the mass balance of a 5 point star
+ *
+ * The mass balance is based on the common groundwater flow equation:
+ *
+ * \f[Ss \frac{\partial h}{\partial t} = \nabla {\bf K} \nabla h + q \f]
+ *
+ * This equation is discretizised with the finite volume method in two dimensions.
+ *
+ * \param gwdata N_gwflow_data2d *
+ * \param geom N_geom_data *
+ * \param col int
+ * \param row int
+ * \return N_data_star *
+ *
+ * */
+N_data_star *N_callback_gwflow_2d(void *gwdata, N_geom_data * geom, int col,
+				  int row)
+{
+    double T_e = 0, T_w = 0, T_n = 0, T_s = 0;
+    double z_e = 0, z_w = 0, z_n = 0, z_s = 0;
+    double dx, dy, Az;
+    double hc_x, hc_y;
+    double z, top;
+    double hc_xw, hc_yn;
+    double z_xw, z_yn;
+    double hc_xe, hc_ys;
+    double z_xe, z_ys;
+    double hc, hc_start;
+    double Ss, r, q;
+    double C, W, E, N, S, V;
+    N_gwflow_data2d *data;
+    N_data_star *mat_pos;
+    double river_vect = 0;	/*entry in vector */
+    double river_mat = 0;	/*entry in matrix */
+    double drain_vect = 0;	/*entry in vector */
+    double drain_mat = 0;	/*entry in matrix */
+
+    /*cast the void pointer to the right data structure */
+    data = (N_gwflow_data2d *) gwdata;
+
+    dx = geom->dx;
+    dy = geom->dy;
+    Az = N_get_geom_data_area_of_cell(geom, row);
+
+    /*read the data from the arrays */
+    hc_start = N_get_array_2d_d_value(data->phead_start, col, row);
+    hc = N_get_array_2d_d_value(data->phead, col, row);
+    top = N_get_array_2d_d_value(data->top, col, row);
+
+    /* Inner sources */
+    q = N_get_array_2d_d_value(data->q, col, row);
+
+    /* storativity or porosity of current cell face [-]*/
+    Ss = N_get_array_2d_d_value(data->s, col, row);
+    /* recharge */
+    r = N_get_array_2d_d_value(data->r, col, row) * Az;
+
+
+    if (hc > top) {		/*If the aquifer is confined */
+	z = N_get_array_2d_d_value(data->top, col,
+				   row) -
+	    N_get_array_2d_d_value(data->bottom, col, row);
+	z_xw =
+	    N_get_array_2d_d_value(data->top, col - 1,
+				   row) -
+	    N_get_array_2d_d_value(data->bottom, col - 1, row);
+	z_xe =
+	    N_get_array_2d_d_value(data->top, col + 1,
+				   row) -
+	    N_get_array_2d_d_value(data->bottom, col + 1, row);
+	z_yn =
+	    N_get_array_2d_d_value(data->top, col,
+				   row - 1) -
+	    N_get_array_2d_d_value(data->bottom, col, row - 1);
+	z_ys =
+	    N_get_array_2d_d_value(data->top, col,
+				   row + 1) -
+	    N_get_array_2d_d_value(data->bottom, col, row + 1);
+    }
+    else {			/* the aquifer is unconfined */
+
+	/* If the aquifer is unconfied use an explicite scheme to solve
+	 * the nonlinear equation. We use the phead from the first iteration */
+	z = N_get_array_2d_d_value(data->phead, col, row) -
+	    N_get_array_2d_d_value(data->bottom, col, row);
+	z_xw = N_get_array_2d_d_value(data->phead, col - 1, row) -
+	    N_get_array_2d_d_value(data->bottom, col - 1, row);
+	z_xe = N_get_array_2d_d_value(data->phead, col + 1, row) -
+	    N_get_array_2d_d_value(data->bottom, col + 1, row);
+	z_yn = N_get_array_2d_d_value(data->phead, col, row - 1) -
+	    N_get_array_2d_d_value(data->bottom, col, row - 1);
+	z_ys = N_get_array_2d_d_value(data->phead, col, row + 1) -
+	    N_get_array_2d_d_value(data->bottom, col, row + 1);
+    }
+
+    /*geometrical mean of cell height */
+    if (z_w > 0 || z_w < 0 || z_w == 0)
+	z_w = N_calc_arith_mean(z_xw, z);
+    else
+	z_w = z;
+    if (z_e > 0 || z_e < 0 || z_e == 0)
+	z_e = N_calc_arith_mean(z_xe, z);
+    else
+	z_e = z;
+    if (z_n > 0 || z_n < 0 || z_n == 0)
+	z_n = N_calc_arith_mean(z_yn, z);
+    else
+	z_n = z;
+    if (z_s > 0 || z_s < 0 || z_s == 0)
+	z_s = N_calc_arith_mean(z_ys, z);
+    else
+	z_s = z;
+
+    /*get the surrounding permeabilities */
+    hc_x = N_get_array_2d_d_value(data->hc_x, col, row);
+    hc_y = N_get_array_2d_d_value(data->hc_y, col, row);
+    hc_xw = N_get_array_2d_d_value(data->hc_x, col - 1, row);
+    hc_xe = N_get_array_2d_d_value(data->hc_x, col + 1, row);
+    hc_yn = N_get_array_2d_d_value(data->hc_y, col, row - 1);
+    hc_ys = N_get_array_2d_d_value(data->hc_y, col, row + 1);
+
+    /* calculate the transmissivities */
+    T_w = N_calc_harmonic_mean(hc_xw, hc_x) * z_w;
+    T_e = N_calc_harmonic_mean(hc_xe, hc_x) * z_e;
+    T_n = N_calc_harmonic_mean(hc_yn, hc_y) * z_n;
+    T_s = N_calc_harmonic_mean(hc_ys, hc_y) * z_s;
+
+    /* Compute the river leakage, this is an explicit method
+     * Influent and effluent flow is computed.
+     */
+    if (data->river_leak &&
+	(N_get_array_2d_d_value(data->river_leak, col, row) != 0) &&
+            N_get_array_2d_d_value(data->river_bed, col, row) <= top) {
+        /* Groundwater surface is above the river bed*/
+	if (hc > N_get_array_2d_d_value(data->river_bed, col, row)) {
+	    river_vect = N_get_array_2d_d_value(data->river_head, col, row) *
+		N_get_array_2d_d_value(data->river_leak, col, row);
+	    river_mat = N_get_array_2d_d_value(data->river_leak, col, row);
+	} /* Groundwater surface is below the river bed */
+	else if (hc < N_get_array_2d_d_value(data->river_bed, col, row)) {
+	    river_vect = (N_get_array_2d_d_value(data->river_head, col, row) -
+			  N_get_array_2d_d_value(data->river_bed, col, row))
+		* N_get_array_2d_d_value(data->river_leak, col, row);
+	    river_mat = 0;
+	}
+    }
+
+    /* compute the drainage, this is an explicit method
+     * Drainage is only enabled, if the drain bed is lower the groundwater surface
+     */
+    if (data->drain_leak &&
+	(N_get_array_2d_d_value(data->drain_leak, col, row) != 0) &&
+            N_get_array_2d_d_value(data->drain_bed, col, row) <= top) {
+	if (hc > N_get_array_2d_d_value(data->drain_bed, col, row)) {
+	    drain_vect = N_get_array_2d_d_value(data->drain_bed, col, row) *
+		N_get_array_2d_d_value(data->drain_leak, col, row);
+	    drain_mat = N_get_array_2d_d_value(data->drain_leak, col, row);
+	}
+	else if (hc <= N_get_array_2d_d_value(data->drain_bed, col, row)) {
+	    drain_vect = 0;
+	    drain_mat = 0;
+	}
+    }
+
+    /*mass balance center cell to western cell */
+    W = -1 * T_w * dy / dx;
+    /*mass balance center cell to eastern cell */
+    E = -1 * T_e * dy / dx;
+    /*mass balance center cell to northern cell */
+    N = -1 * T_n * dx / dy;
+    /*mass balance center cell to southern cell */
+    S = -1 * T_s * dx / dy;
+
+    /*the diagonal entry of the matrix */
+    C = -1 * (W + E + N + S -  Az *Ss / data->dt - river_mat * Az -
+	      drain_mat * Az);
+
+    /*the entry in the right side b of Ax = b */
+    V = (q + hc_start * Az * Ss / data->dt) + r + river_vect * Az +
+	drain_vect * Az;
+
+    G_debug(5, "N_callback_gwflow_2d: called [%i][%i]", row, col);
+
+    /*create the 5 point star entries */
+    mat_pos = N_create_5star(C, W, E, N, S, V);
+
+    return mat_pos;
+}
+
+
+
+/* *************************************************************** */
+/* ****************** N_gwflow_2d_calc_water_budget ************** */
+/* *************************************************************** */
+/*!
+ * \brief This function computes the water budget of the entire groundwater
+ *
+ * The water budget is calculated for each active and dirichlet cell from
+ * its surrounding neighbours. This is based on the 5 star mass balance computation
+ * of N_callback_gwflow_2d and the gradient of the water heights in the cells.
+ * The sum of the water budget of each active/dirichlet cell must be near zero
+ * due the effect of numerical inaccuracy of cpu's.
+ *
+ * \param gwdata N_gwflow_data2d *
+ * \param geom N_geom_data *
+ * \param budget N_array_2d
+ * \return void
+ *
+ * */
+void
+N_gwflow_2d_calc_water_budget(N_gwflow_data2d * data, N_geom_data * geom, N_array_2d * budget)
+{
+    int y, x, stat;
+    double h, hc;
+    double val;
+    double sum;
+    N_data_star *dstar;
+
+    int rows = data->status->rows;
+    int cols = data->status->cols;
+
+    sum = 0;
+
+    for (y = 0; y < rows; y++) {
+	G_percent(y, rows - 1, 10);
+	for (x = 0; x < cols; x++) {
+	    stat = N_get_array_2d_c_value(data->status, x, y);
+
+            val = 0.0;
+
+	    if (stat != N_CELL_INACTIVE ) {	/*all active/dirichlet cells */
+
+                /* Compute the flow parameter */
+                dstar = N_callback_gwflow_2d(data, geom, x, y);
+                /* Compute the gradient in each direction pointing from the center */
+                hc = N_get_array_2d_d_value(data->phead, x, y);
+
+                if((int)N_get_array_2d_d_value(data->status, x + 1, y    ) != N_CELL_INACTIVE) {
+                    h = N_get_array_2d_d_value(data->phead,  x + 1, y);
+                    val += dstar->E * (hc - h);
+                }
+                if((int)N_get_array_2d_d_value(data->status, x - 1, y    ) != N_CELL_INACTIVE) {
+                    h = N_get_array_2d_d_value(data->phead,  x - 1, y);
+                    val += dstar->W * (hc - h);
+                }
+                if((int)N_get_array_2d_d_value(data->status, x    , y + 1) != N_CELL_INACTIVE) {
+                    h = N_get_array_2d_d_value(data->phead,  x    , y + 1);
+                    val += dstar->S * (hc - h);
+                }
+                if((int)N_get_array_2d_d_value(data->status, x    , y - 1) != N_CELL_INACTIVE) {
+                    h = N_get_array_2d_d_value(data->phead,  x    , y - 1);
+                    val += dstar->N * (hc - h);
+                }
+
+                sum += val;
+
+                G_free(dstar);
+	    }
+	    else {
+		Rast_set_null_value(&val, 1, DCELL_TYPE);
+	    }
+	    N_put_array_2d_d_value(budget, x, y, val);
+	}
+    }
+
+    if(fabs(sum) < 0.0000000001)
+        G_message(_("The total sum of the water budget: %g\n"), sum);
+    else
+        G_warning(_("The total sum of the water budget is significantly larger then 0: %g\n"), sum);
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_heatflow.c (from rev 62429, grass/trunk/lib/gpde/N_heatflow.c)
===================================================================
--- grass/trunk/lib/gpde/n_heatflow.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_heatflow.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,19 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      Calculation of heatflow
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <grass/N_pde.h>

Copied: grass/trunk/lib/gpde/n_les.c (from rev 62429, grass/trunk/lib/gpde/N_les.c)
===================================================================
--- grass/trunk/lib/gpde/n_les.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_les.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,335 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      functions to manage linear equation systems
+* 		part of the gpde library
+*               
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <stdlib.h>
+#include <grass/N_pde.h>
+#include <grass/gmath.h>
+
+
+/*!
+ * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A, vector x and vector b
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param cols int
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_nquad_les(int cols, int rows, int type)
+{
+    return N_alloc_les_param(cols, rows, type, 2);
+}
+
+/*!
+ * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A and vector x
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param cols int
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_nquad_les_Ax(int cols, int rows, int type)
+{
+    return N_alloc_les_param(cols, rows, type, 1);
+}
+
+/*!
+ * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param cols int
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_nquad_les_A(int cols, int rows, int type)
+{
+    return N_alloc_les_param(cols, rows, type, 0);
+}
+
+/*!
+ * \brief Allocate memory for a (not) quadratic linear equation system which includes the Matrix A, vector x and vector b
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param cols int
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_nquad_les_Ax_b(int cols, int rows, int type)
+{
+    return N_alloc_les_param(cols, rows, type, 2);
+}
+
+
+
+/*!
+ * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A, vector x and vector b
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_les(int rows, int type)
+{
+    return N_alloc_les_param(rows, rows, type, 2);
+}
+
+/*!
+ * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A and vector x
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_les_Ax(int rows, int type)
+{
+    return N_alloc_les_param(rows, rows, type, 1);
+}
+
+/*!
+ * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_les_A(int rows, int type)
+{
+    return N_alloc_les_param(rows, rows, type, 0);
+}
+
+/*!
+ * \brief Allocate memory for a quadratic linear equation system which includes the Matrix A, vector x and vector b
+ *
+ * This function calls #N_alloc_les_param
+ *
+ * \param rows int
+ * \param type int
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_les_Ax_b(int rows, int type)
+{
+    return N_alloc_les_param(rows, rows, type, 2);
+}
+
+
+/*!
+ * \brief Allocate memory for a quadratic or not quadratic linear equation system
+ *
+ * The type of the linear equation system must be N_NORMAL_LES for
+ * a regular quadratic matrix or N_SPARSE_LES for a sparse matrix
+ *
+ * <p>
+ * In case of N_NORMAL_LES
+ * 
+ * A quadratic matrix of size rows*rows*sizeof(double) will allocated
+ *
+ * <p>
+ * In case of N_SPARSE_LES
+ *
+ * a vector of size row will be allocated, ready to hold additional allocated sparse vectors.
+ * each sparse vector may have a different size.
+ *
+ * Parameter parts defines which parts of the les should be allocated.
+ * The number of columns and rows defines if the matrix is quadratic.
+ *
+ * \param cols int
+ * \param rows int
+ * \param type int
+ * \param parts int -- 2 = A, x and b; 1 = A and x; 0 = A allocated
+ * \return N_les *
+ *
+ * */
+N_les *N_alloc_les_param(int cols, int rows, int type, int parts)
+{
+    N_les *les;
+
+    int i;
+
+    if (type == N_SPARSE_LES)
+	G_debug(2,
+		"Allocate memory for a sparse linear equation system with %i rows\n",
+		rows);
+    else
+	G_debug(2,
+		"Allocate memory for a regular linear equation system with %i rows\n",
+		rows);
+
+    les = (N_les *) G_calloc(1, sizeof(N_les));
+
+    if (parts > 0) {
+	les->x = (double *)G_calloc(cols, sizeof(double));
+	for (i = 0; i < cols; i++)
+	    les->x[i] = 0.0;
+    }
+
+
+    if (parts > 1) {
+	les->b = (double *)G_calloc(cols, sizeof(double));
+	for (i = 0; i < cols; i++)
+	    les->b[i] = 0.0;
+    }
+
+    les->A = NULL;
+    les->Asp = NULL;
+    les->rows = rows;
+    les->cols = cols;
+    if (rows == cols)
+	les->quad = 1;
+    else
+	les->quad = 0;
+
+    if (type == N_SPARSE_LES) {
+	les->Asp = G_math_alloc_spmatrix(rows);
+	les->type = N_SPARSE_LES;
+    }
+    else {
+	les->A = G_alloc_matrix(rows, cols);
+	les->type = N_NORMAL_LES;
+    }
+
+    return les;
+}
+
+/*!
+ *
+ * \brief prints the linear equation system to stdout
+ *
+ * <p>
+ * Format:
+ * A*x = b
+ *
+ * <p>
+ * Example
+ \verbatim
+
+ 2 1 1 1 * 2 = 0.1
+ 1 2 0 0 * 3 = 0.2
+ 1 0 2 0 * 3 = 0.2
+ 1 0 0 2 * 2 = 0.1
+
+ \endverbatim
+ *
+ * \param les N_les * 
+ * \return void
+ *  
+ * */
+void N_print_les(N_les * les)
+{
+    int i, j, k, out;
+
+
+    if (les->type == N_SPARSE_LES) {
+	for (i = 0; i < les->rows; i++) {
+	    for (j = 0; j < les->cols; j++) {
+		out = 0;
+		for (k = 0; k < les->Asp[i]->cols; k++) {
+		    if (les->Asp[i]->index[k] == j) {
+			fprintf(stdout, "%4.5f ", les->Asp[i]->values[k]);
+			out = 1;
+		    }
+		}
+		if (!out)
+		    fprintf(stdout, "%4.5f ", 0.0);
+	    }
+	    if (les->x)
+		fprintf(stdout, "  *  %4.5f", les->x[i]);
+	    if (les->b)
+		fprintf(stdout, " =  %4.5f ", les->b[i]);
+
+	    fprintf(stdout, "\n");
+	}
+    }
+    else {
+
+	for (i = 0; i < les->rows; i++) {
+	    for (j = 0; j < les->cols; j++) {
+		fprintf(stdout, "%4.5f ", les->A[i][j]);
+	    }
+	    if (les->x)
+		fprintf(stdout, "  *  %4.5f", les->x[i]);
+	    if (les->b)
+		fprintf(stdout, " =  %4.5f ", les->b[i]);
+
+	    fprintf(stdout, "\n");
+	}
+
+    }
+    return;
+}
+
+/*!
+ * \brief Release the memory of the linear equation system
+ *
+ * \param les N_les *            
+ * \return void
+ *
+ * */
+
+void N_free_les(N_les * les)
+{
+    if (les->type == N_SPARSE_LES)
+	G_debug(2, "Releasing memory of a sparse linear equation system\n");
+    else
+	G_debug(2, "Releasing memory of a regular linear equation system\n");
+
+    if (les) {
+
+	if (les->x)
+	    G_free(les->x);
+	if (les->b)
+	    G_free(les->b);
+
+	if (les->type == N_SPARSE_LES) {
+
+	    if (les->Asp) {
+		G_math_free_spmatrix(les->Asp, les->rows);
+	    }
+	}
+	else {
+
+	    if (les->A) {
+		G_free_matrix(les->A);
+	    }
+	}
+
+	free(les);
+    }
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_les_assemble.c (from rev 62429, grass/trunk/lib/gpde/N_les_assemble.c)
===================================================================
--- grass/trunk/lib/gpde/n_les_assemble.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_les_assemble.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,1397 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      functions to assemble a linear equation system
+* 		part of the gpde library
+*               
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+
+#include <math.h>
+#include <grass/N_pde.h>
+
+/* local protos */
+static int make_les_entry_2d(int i, int j, int offset_i, int offset_j,
+			     int count, int pos, N_les * les,
+			     G_math_spvector * spvect,
+			     N_array_2d * cell_count, N_array_2d * status,
+			     N_array_2d * start_val, double entry,
+			     int cell_type);
+
+static int make_les_entry_3d(int i, int j, int k, int offset_i, int offset_j,
+			     int offset_k, int count, int pos, N_les * les,
+			     G_math_spvector * spvect,
+			     N_array_3d * cell_count, N_array_3d * status,
+			     N_array_3d * start_val, double entry,
+			     int cell_type);
+
+/* *************************************************************** * 
+ * ********************** N_alloc_5star ************************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate a 5 point star data structure
+ *
+ * \return N_data_star *
+ * */
+N_data_star *N_alloc_5star(void)
+{
+    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
+
+    star->type = N_5_POINT_STAR;
+    star->count = 5;
+    return star;
+}
+
+/* *************************************************************** * 
+ * ********************* N_alloc_7star *************************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate a 7 point star data structure
+ *
+ * \return N_data_star *
+ * */
+N_data_star *N_alloc_7star(void)
+{
+    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
+
+    star->type = N_7_POINT_STAR;
+    star->count = 7;
+    return star;
+}
+
+/* *************************************************************** * 
+ * ********************* N_alloc_9star *************************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate a 9 point star data structure
+ *
+ * \return N_data_star *
+ *
+ * \attention The 9 point start is not yet implemented in the matrix assembling function
+ *
+ * */
+N_data_star *N_alloc_9star(void)
+{
+    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
+
+    star->type = N_9_POINT_STAR;
+    star->count = 9;
+    return star;
+}
+
+/* *************************************************************** * 
+ * ********************* N_alloc_27star ************************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate a 27 point star data structure
+ *
+ * \return N_data_star *
+ *
+ * \attention The 27 point start is not yet implemented in the matrix assembling function
+ *
+ * */
+N_data_star *N_alloc_27star(void)
+{
+    N_data_star *star = (N_data_star *) G_calloc(1, sizeof(N_data_star));
+
+    star->type = N_27_POINT_STAR;
+    star->count = 27;
+    return star;
+}
+
+/* *************************************************************** * 
+ * ********************** N_create_5star ************************* * 
+ * *************************************************************** */
+/*!
+ * \brief allocate and initialize a 5 point star data structure
+ *
+ * \param C double
+ * \param W double
+ * \param E double
+ * \param N double
+ * \param S double
+ * \param V double
+ * \return N_data_star *
+ * */
+N_data_star *N_create_5star(double C, double W, double E, double N,
+			    double S, double V)
+{
+    N_data_star *star = N_alloc_5star();
+
+    star->C = C;
+    star->W = W;
+    star->E = E;
+    star->N = N;
+    star->S = S;
+
+    star->V = V;
+
+    G_debug(5, "N_create_5star:  w %g e %g n %g s %g c %g v %g\n", star->W,
+	    star->E, star->N, star->S, star->C, star->V);
+
+    return star;
+}
+
+/* *************************************************************** * 
+ * ************************* N_create_7star ********************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate and initialize a 7 point star data structure
+ *
+ * \param C double
+ * \param W double
+ * \param E double
+ * \param N double
+ * \param S double
+ * \param T double
+ * \param B double
+ * \param V double
+ * \return N_data_star *
+ * */
+N_data_star *N_create_7star(double C, double W, double E, double N,
+			    double S, double T, double B, double V)
+{
+    N_data_star *star = N_alloc_7star();
+
+    star->C = C;
+    star->W = W;
+    star->E = E;
+    star->N = N;
+    star->S = S;
+
+    star->T = T;
+    star->B = B;
+
+    star->V = V;
+
+    G_debug(5, "N_create_7star:  w %g e %g n %g s %g t %g b %g c %g v %g\n",
+	    star->W, star->E, star->N, star->S, star->T, star->B, star->C,
+	    star->V);
+
+    return star;
+}
+
+/* *************************************************************** * 
+ * ************************ N_create_9star *********************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate and initialize a 9 point star data structure
+ *
+ * \param C  double
+ * \param W  double
+ * \param E  double
+ * \param N  double
+ * \param S  double
+ * \param NW double
+ * \param SW double
+ * \param NE double
+ * \param SE double
+ * \param V  double
+ * \return N_data_star *
+ * */
+N_data_star *N_create_9star(double C, double W, double E, double N,
+			    double S, double NW, double SW, double NE,
+			    double SE, double V)
+{
+    N_data_star *star = N_alloc_9star();
+
+    star->C = C;
+    star->W = W;
+    star->E = E;
+    star->N = N;
+    star->S = S;
+
+    star->NW = NW;
+    star->SW = SW;
+    star->NE = NE;
+    star->SE = SE;
+
+    star->V = V;
+
+    G_debug(5,
+	    "N_create_9star:  w %g e %g n %g s %g nw %g sw %g ne %g se %g c %g v %g\n",
+	    star->W, star->E, star->N, star->S, star->NW, star->SW, star->NE,
+	    star->SE, star->C, star->V);
+
+    return star;
+}
+
+/* *************************************************************** * 
+ * ************************ N_create_27star *********************** * 
+ * *************************************************************** */
+/*!
+ * \brief allocate and initialize a 27 point star data structure
+ *
+ * \param C  double
+ * \param W  double
+ * \param E  double
+ * \param N  double
+ * \param S  double
+ * \param NW double
+ * \param SW double
+ * \param NE double
+ * \param SE double
+ * \param T  double
+ * \param W_T  double
+ * \param E_T  double
+ * \param N_T  double
+ * \param S_T  double
+ * \param NW_T double
+ * \param SW_T double
+ * \param NE_T double
+ * \param SE_T double
+ * \param B  double
+ * \param W_B  double
+ * \param E_B  double
+ * \param N_B  double
+ * \param S_B  double
+ * \param NW_B double
+ * \param SW_B double
+ * \param NE_B double
+ * \param SE_B double
+ * \param V  double
+ * \return N_data_star *
+ * */
+N_data_star *N_create_27star(double C, double W, double E, double N, double S,
+			     double NW, double SW, double NE, double SE,
+			     double T, double W_T, double E_T, double N_T,
+			     double S_T, double NW_T, double SW_T,
+			     double NE_T, double SE_T, double B, double W_B,
+			     double E_B, double N_B, double S_B, double NW_B,
+			     double SW_B, double NE_B, double SE_B, double V)
+{
+    N_data_star *star = N_alloc_27star();
+
+    star->C = C;
+    star->W = W;
+    star->E = E;
+    star->N = N;
+    star->S = S;
+
+    star->NW = NW;
+    star->SW = SW;
+    star->NE = NE;
+    star->SE = SE;
+
+    star->T = T;
+    star->W_T = W_T;
+    star->E_T = E_T;
+    star->N_T = N_T;
+    star->S_T = S_T;
+
+    star->NW_T = NW_T;
+    star->SW_T = SW_T;
+    star->NE_T = NE_T;
+    star->SE_T = SE_T;
+
+    star->B = B;
+    star->W_B = W_B;
+    star->E_B = E_B;
+    star->N_B = N_B;
+    star->S_B = S_B;
+
+    star->NW_B = NW_B;
+    star->SW_B = SW_B;
+    star->NE_B = NE_B;
+    star->SE_B = SE_B;
+
+    star->V = V;
+
+    G_debug(5,
+	    "N_create_27star:  w %g e %g n %g s %g nw %g sw %g ne %g se %g c %g v %g\n",
+	    star->W, star->E, star->N, star->S, star->NW, star->SW, star->NE,
+	    star->SE, star->C, star->V);
+
+    G_debug(5,
+	    "N_create_27star:  w_t %g e_t %g n_t %g s_t %g nw_t %g sw_t %g ne_t %g se_t %g t %g \n",
+	    star->W_T, star->E_T, star->N_T, star->S_T, star->NW_T,
+	    star->SW_T, star->NE_T, star->SE_T, star->T);
+
+    G_debug(5,
+	    "N_create_27star:  w_b %g e_b %g n_b %g s_b %g nw_b %g sw_b %g ne_b %g se_B %g b %g\n",
+	    star->W_B, star->E_B, star->N_B, star->S_B, star->NW_B,
+	    star->SW_B, star->NE_B, star->SE_B, star->B);
+
+
+
+    return star;
+}
+
+
+/* *************************************************************** * 
+ * ****************** N_set_les_callback_3d_func ***************** * 
+ * *************************************************************** */
+/*!
+ * \brief Set the callback function which is called while assembling the les in 3d
+ *
+ * \param data N_les_callback_3d *
+ * \param callback_func_3d N_data_star *
+ * \return void
+ * */
+void
+N_set_les_callback_3d_func(N_les_callback_3d * data,
+			   N_data_star * (*callback_func_3d) ())
+{
+    data->callback = callback_func_3d;
+}
+
+/* *************************************************************** * 
+ * *************** N_set_les_callback_2d_func ******************** * 
+ * *************************************************************** */
+/*!
+ * \brief Set the callback function which is called while assembling the les in 2d
+ *
+ * \param data N_les_callback_2d *
+ * \param callback_func_2d N_data_star * 
+ * \return void
+ * */
+void
+N_set_les_callback_2d_func(N_les_callback_2d * data,
+			   N_data_star * (*callback_func_2d) ())
+{
+    data->callback = callback_func_2d;
+}
+
+/* *************************************************************** * 
+ * ************** N_alloc_les_callback_3d ************************ * 
+ * *************************************************************** */
+/*!
+ * \brief Allocate the structure holding the callback function
+ *
+ * A template callback is set. Use N_set_les_callback_3d_func
+ * to set up a specific function.
+ *
+ * \return N_les_callback_3d *
+ * */
+N_les_callback_3d *N_alloc_les_callback_3d(void)
+{
+    N_les_callback_3d *call;
+
+    call = (N_les_callback_3d *) G_calloc(1, sizeof(N_les_callback_3d *));
+    call->callback = N_callback_template_3d;
+
+    return call;
+}
+
+/* *************************************************************** * 
+ * *************** N_alloc_les_callback_2d *********************** * 
+ * *************************************************************** */
+/*!
+ * \brief Allocate the structure holding the callback function
+ *
+ * A template callback is set. Use N_set_les_callback_2d_func
+ * to set up a specific function.
+ *
+ * \return N_les_callback_2d *
+ * */
+N_les_callback_2d *N_alloc_les_callback_2d(void)
+{
+    N_les_callback_2d *call;
+
+    call = (N_les_callback_2d *) G_calloc(1, sizeof(N_les_callback_2d *));
+    call->callback = N_callback_template_2d;
+
+    return call;
+}
+
+/* *************************************************************** * 
+ * ******************** N_callback_template_3d ******************* * 
+ * *************************************************************** */
+/*!
+ * \brief A callback template creates a 7 point star structure
+ *
+ * This is a template callback for mass balance calculation with 7 point stars
+ * based on 3d data (g3d).
+ *
+ * \param data void *
+ * \param geom N_geom_data *
+ * \param depth int
+ * \param row   int
+ * \param col   int
+ * \return N_data_star *
+ *
+ * */
+N_data_star *N_callback_template_3d(void *data, N_geom_data * geom, int col,
+				    int row, int depth)
+{
+    N_data_star *star = N_alloc_7star();
+
+    star->E = 1 / geom->dx;
+    star->W = 1 / geom->dx;
+    star->N = 1 / geom->dy;
+    star->S = 1 / geom->dy;
+    star->T = 1 / geom->dz;
+    star->B = 1 / geom->dz;
+    star->C = -1 * (2 / geom->dx + 2 / geom->dy + 2 / geom->dz);
+    star->V = -1;
+
+    G_debug(5,
+	    "N_callback_template_3d:  w %g e %g n %g s %g t %g b %g c %g v %g\n",
+	    star->W, star->E, star->N, star->S, star->T, star->B, star->C,
+	    star->V);
+
+
+    return star;
+}
+
+/* *************************************************************** * 
+ * ********************* N_callback_template_2d ****************** * 
+ * *************************************************************** */
+/*!
+ * \brief A callback template creates a 9 point star structure
+ *
+ * This is a template callback for mass balance calculation with 9 point stars
+ * based on 2d data (raster).
+ *
+ * \param data void *
+ * \param geom N_geom_data *
+ * \param row int
+ * \param col int
+ * \return N_data_star *
+ *
+ * */
+N_data_star *N_callback_template_2d(void *data, N_geom_data * geom, int col,
+				    int row)
+{
+    N_data_star *star = N_alloc_9star();
+
+    star->E = 1 / geom->dx;
+    star->NE = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
+    star->SE = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
+    star->W = 1 / geom->dx;
+    star->NW = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
+    star->SW = 1 / sqrt(geom->dx * geom->dx + geom->dy * geom->dy);
+    star->N = 1 / geom->dy;
+    star->S = 1 / geom->dy;
+    star->C =
+	-1 * (star->E + star->NE + star->SE + star->W + star->NW + star->SW +
+	      star->N + star->S);
+    star->V = 0;
+
+    return star;
+}
+
+/* *************************************************************** * 
+ * ******************** N_assemble_les_2d ************************ * 
+ * *************************************************************** */
+/*!
+ * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active cells
+ *
+ * This function calls #N_assemble_les_2d_param
+ *
+ */
+N_les *N_assemble_les_2d(int les_type, N_geom_data * geom,
+			 N_array_2d * status, N_array_2d * start_val,
+			 void *data, N_les_callback_2d * call)
+{
+    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_ACTIVE);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active cells
+ *
+ * This function calls #N_assemble_les_2d_param
+ *
+ */
+N_les *N_assemble_les_2d_active(int les_type, N_geom_data * geom,
+				N_array_2d * status, N_array_2d * start_val,
+				void *data, N_les_callback_2d * call)
+{
+    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_ACTIVE);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 2d location data (raster) and active and dirichlet cells
+ *
+ * This function calls #N_assemble_les_2d_param
+ *
+ */
+N_les *N_assemble_les_2d_dirichlet(int les_type, N_geom_data * geom,
+				   N_array_2d * status,
+				   N_array_2d * start_val, void *data,
+				   N_les_callback_2d * call)
+{
+    return N_assemble_les_2d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_DIRICHLET);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 2d location data  (raster)
+ *
+ * 
+ * The linear equation system type can be set to N_NORMAL_LES to create a regular
+ * matrix, or to N_SPARSE_LES to create a sparse matrix. This function returns
+ * a new created linear equation system which can be solved with 
+ * linear equation solvers. An 2d array with start values and an 2d status array
+ * must be provided as well as the location geometry and a void pointer to data 
+ * passed to the callback which creates the les row entries. This callback
+ * must be defined in the N_les_callback_2d strcuture.
+ *
+ * The creation of the les is parallelized with OpenMP. 
+ * If you implement new callbacks, please make sure that the 
+ * function calls are thread safe.
+ *
+ *
+ * the les can be created in two ways, with dirichlet and similar cells and without them,
+ * to spare some memory. If the les is created with dirichlet cell, the dirichlet boundary condition
+ * must be added.
+ *
+ * \param les_type int
+ * \param geom      N_geom_data*
+ * \param status    N_array_2d *
+ * \param start_val N_array_2d *
+ * \param data void *
+ * \param cell_type int  -- les assemble based on N_CELL_ACTIVE or N_CELL_DIRICHLET
+ * \param call N_les_callback_2d *
+ * \return N_les *
+ * */
+N_les *N_assemble_les_2d_param(int les_type, N_geom_data * geom,
+			       N_array_2d * status, N_array_2d * start_val,
+			       void *data, N_les_callback_2d * call,
+			       int cell_type)
+{
+    int i, j, count = 0, pos = 0;
+    int cell_type_count = 0;
+    int **index_ij;
+    N_array_2d *cell_count;
+    N_les *les = NULL;
+
+    G_debug(2,
+	    "N_assemble_les_2d: starting to assemble the linear equation system");
+
+    /* At first count the number of valid cells and save 
+     * each number in a new 2d array. Those numbers are used 
+     * to create the linear equation system.
+     * */
+
+    cell_count = N_alloc_array_2d(geom->cols, geom->rows, 1, CELL_TYPE);
+
+    /* include dirichlet cells in the les */
+    if (cell_type == N_CELL_DIRICHLET) {
+	for (j = 0; j < geom->rows; j++) {
+	    for (i = 0; i < geom->cols; i++) {
+		/*use all non-inactive cells for les creation */
+		if (N_CELL_INACTIVE < N_get_array_2d_c_value(status, i, j) &&
+		    N_get_array_2d_c_value(status, i, j) < N_MAX_CELL_STATE)
+		    cell_type_count++;
+	    }
+	}
+    }
+    /*use only active cell in the les */
+    if (cell_type == N_CELL_ACTIVE) {
+	for (j = 0; j < geom->rows; j++) {
+	    for (i = 0; i < geom->cols; i++) {
+		/*count only active cells */
+		if (N_CELL_ACTIVE == N_get_array_2d_d_value(status, i, j))
+		    cell_type_count++;
+	    }
+	}
+    }
+
+    G_debug(2, "N_assemble_les_2d: number of used cells %i\n",
+	    cell_type_count);
+
+    if (cell_type_count == 0)
+	G_fatal_error
+	    ("Not enough cells [%i] to create the linear equation system. Check the cell status. Only active cells (value = 1) are used to create the equation system.",
+	     cell_type_count);
+
+    /* Then allocate the memory for the linear equation system (les). 
+     * Only valid cells are used to create the les. */
+    index_ij = (int **)G_calloc(cell_type_count, sizeof(int *));
+    for (i = 0; i < cell_type_count; i++)
+	index_ij[i] = (int *)G_calloc(2, sizeof(int));
+
+    les = N_alloc_les_Ax_b(cell_type_count, les_type);
+
+    count = 0;
+
+    /*count the number of cells which should be used to create the linear equation system */
+    /*save the i and j indices and create a ordered numbering */
+    for (j = 0; j < geom->rows; j++) {
+	for (i = 0; i < geom->cols; i++) {
+	    /*count every non-inactive cell */
+	    if (cell_type == N_CELL_DIRICHLET) {
+		if (N_CELL_INACTIVE < N_get_array_2d_c_value(status, i, j) &&
+		    N_get_array_2d_c_value(status, i, j) < N_MAX_CELL_STATE) {
+		    N_put_array_2d_c_value(cell_count, i, j, count);
+		    index_ij[count][0] = i;
+		    index_ij[count][1] = j;
+		    count++;
+		    G_debug(5,
+			    "N_assemble_les_2d: non-inactive cells count %i at pos x[%i] y[%i]\n",
+			    count, i, j);
+		}
+		/*count every active cell */
+	    }
+	    else if (N_CELL_ACTIVE == N_get_array_2d_c_value(status, i, j)) {
+		N_put_array_2d_c_value(cell_count, i, j, count);
+		index_ij[count][0] = i;
+		index_ij[count][1] = j;
+		count++;
+		G_debug(5,
+			"N_assemble_les_2d: active cells count %i at pos x[%i] y[%i]\n",
+			count, i, j);
+	    }
+	}
+    }
+
+    G_debug(2, "N_assemble_les_2d: starting the parallel assemble loop");
+
+    /* Assemble the matrix in parallel */
+#pragma omp parallel for private(i, j, pos, count) schedule(static)
+    for (count = 0; count < cell_type_count; count++) {
+	i = index_ij[count][0];
+	j = index_ij[count][1];
+
+	/*create the entries for the */
+	N_data_star *items = call->callback(data, geom, i, j);
+
+	/* we need a sparse vector pointer anytime */
+	G_math_spvector *spvect = NULL;
+
+	/*allocate a sprase vector */
+	if (les_type == N_SPARSE_LES) {
+	    spvect = G_math_alloc_spvector(items->count);
+	}
+	/* initial conditions */
+	les->x[count] = N_get_array_2d_d_value(start_val, i, j);
+
+	/* the entry in the vector b */
+	les->b[count] = items->V;
+
+	/* pos describes the position in the sparse vector.
+	 * the first entry is always the diagonal entry of the matrix*/
+	pos = 0;
+
+	if (les_type == N_SPARSE_LES) {
+	    spvect->index[pos] = count;
+	    spvect->values[pos] = items->C;
+	}
+	else {
+	    les->A[count][count] = items->C;
+	}
+	/* western neighbour, entry is col - 1 */
+	if (i > 0) {
+	    pos = make_les_entry_2d(i, j, -1, 0, count, pos, les, spvect,
+				    cell_count, status, start_val, items->W,
+				    cell_type);
+	}
+	/* eastern neighbour, entry col + 1 */
+	if (i < geom->cols - 1) {
+	    pos = make_les_entry_2d(i, j, 1, 0, count, pos, les, spvect,
+				    cell_count, status, start_val, items->E,
+				    cell_type);
+	}
+	/* northern neighbour, entry row - 1 */
+	if (j > 0) {
+	    pos =
+		make_les_entry_2d(i, j, 0, -1, count, pos, les, spvect,
+				  cell_count, status, start_val, items->N,
+				  cell_type);
+	}
+	/* southern neighbour, entry row + 1 */
+	if (j < geom->rows - 1) {
+	    pos = make_les_entry_2d(i, j, 0, 1, count, pos, les, spvect,
+				    cell_count, status, start_val, items->S,
+				    cell_type);
+	}
+	/*in case of a nine point star, we have additional entries */
+	if (items->type == N_9_POINT_STAR) {
+	    /* north-western neighbour, entry is col - 1 row - 1 */
+	    if (i > 0 && j > 0) {
+		pos = make_les_entry_2d(i, j, -1, -1, count, pos, les, spvect,
+					cell_count, status, start_val,
+					items->NW, cell_type);
+	    }
+	    /* north-eastern neighbour, entry col + 1 row - 1 */
+	    if (i < geom->cols - 1 && j > 0) {
+		pos = make_les_entry_2d(i, j, 1, -1, count, pos, les, spvect,
+					cell_count, status, start_val,
+					items->NE, cell_type);
+	    }
+	    /* south-western neighbour, entry is col - 1 row + 1 */
+	    if (i > 0 && j < geom->rows - 1) {
+		pos = make_les_entry_2d(i, j, -1, 1, count, pos, les, spvect,
+					cell_count, status, start_val,
+					items->SW, cell_type);
+	    }
+	    /* south-eastern neighbour, entry col + 1 row + 1 */
+	    if (i < geom->cols - 1 && j < geom->rows - 1) {
+		pos = make_les_entry_2d(i, j, 1, 1, count, pos, les, spvect,
+					cell_count, status, start_val,
+					items->SE, cell_type);
+	    }
+	}
+
+	/*How many entries in the les */
+	if (les->type == N_SPARSE_LES) {
+	    spvect->cols = pos + 1;
+	    G_math_add_spvector(les->Asp, spvect, count);
+	}
+
+	if (items)
+	    G_free(items);
+    }
+
+    /*release memory */
+    N_free_array_2d(cell_count);
+
+    for (i = 0; i < cell_type_count; i++)
+	G_free(index_ij[i]);
+
+    G_free(index_ij);
+
+    return les;
+}
+
+/*!
+ * \brief Integrate Dirichlet or Transmission boundary conditions into the les (2s)
+ *
+ * Dirichlet and Transmission boundary conditions will be integrated into
+ * the provided linear equation system. This is meaningfull if
+ * the les was created with #N_assemble_les_2d_dirichlet, because in
+ * this case Dirichlet boundary conditions are not automatically included.
+ *
+ * The provided les will be modified:
+ *
+ * Ax = b will be splitted into Ax_u + Ax_d = b
+ *
+ * x_u - the unknowns
+ * x_d - the Dirichlet cells
+ *
+ * Ax_u = b -Ax_d will be computed. Then the matrix A will be modified to
+ *
+ * | A_u  0 | x_u
+ * |  0   I | x_d
+ *
+ * \param les N_les* -- the linear equation system
+ * \param geom N_geom_data* -- geometrical data information
+ * \param status N_array_2d* -- the status array containing the cell types
+ * \param start_val N_array_2d* -- an array with start values
+ * \return int -- 1 = success, 0 = failure
+ * */
+int N_les_integrate_dirichlet_2d(N_les * les, N_geom_data * geom,
+				 N_array_2d * status, N_array_2d * start_val)
+{
+    int rows, cols;
+    int count = 0;
+    int i, j, x, y, stat;
+    double *dvect1;
+    double *dvect2;
+
+    G_debug(2,
+	    "N_les_integrate_dirichlet_2d: integrating the dirichlet boundary condition");
+
+    rows = geom->rows;
+    cols = geom->cols;
+
+    /*we nned to additional vectors */
+    dvect1 = (double *)G_calloc(les->cols, sizeof(double));
+    dvect2 = (double *)G_calloc(les->cols, sizeof(double));
+
+    /*fill the first one with the x vector data of Dirichlet cells */
+    count = 0;
+    for (y = 0; y < rows; y++) {
+	for (x = 0; x < cols; x++) {
+	    stat = N_get_array_2d_c_value(status, x, y);
+	    if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
+		dvect1[count] = N_get_array_2d_d_value(start_val, x, y);
+		count++;
+	    }
+	    else if (stat == N_CELL_ACTIVE) {
+		dvect1[count] = 0.0;
+		count++;
+	    }
+	}
+    }
+
+#pragma omp parallel default(shared)
+    {
+	/*perform the matrix vector product and */
+	if (les->type == N_SPARSE_LES)
+	    G_math_Ax_sparse(les->Asp, dvect1, dvect2, les->rows);
+	else
+	    G_math_d_Ax(les->A, dvect1, dvect2, les->rows, les->cols);
+#pragma omp for schedule (static) private(i)
+	for (i = 0; i < les->cols; i++)
+	    les->b[i] = les->b[i] - dvect2[i];
+    }
+
+    /*now set the Dirichlet cell rows and cols to zero and the 
+     * diagonal entry to 1*/
+    count = 0;
+    for (y = 0; y < rows; y++) {
+	for (x = 0; x < cols; x++) {
+	    stat = N_get_array_2d_c_value(status, x, y);
+	    if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
+		if (les->type == N_SPARSE_LES) {
+		    /*set the rows to zero */
+		    for (i = 0; i < les->Asp[count]->cols; i++)
+			les->Asp[count]->values[i] = 0.0;
+		    /*set the cols to zero */
+		    for (i = 0; i < les->rows; i++) {
+			for (j = 0; j < les->Asp[i]->cols; j++) {
+			    if (les->Asp[i]->index[j] == count)
+				les->Asp[i]->values[j] = 0.0;
+			}
+		    }
+
+		    /*entry on the diagonal */
+		    les->Asp[count]->values[0] = 1.0;
+
+		}
+		else {
+		    /*set the rows to zero */
+		    for (i = 0; i < les->cols; i++)
+			les->A[count][i] = 0.0;
+		    /*set the cols to zero */
+		    for (i = 0; i < les->rows; i++)
+			les->A[i][count] = 0.0;
+
+		    /*entry on the diagonal */
+		    les->A[count][count] = 1.0;
+		}
+	    }
+	    if (stat >= N_CELL_ACTIVE)
+		count++;
+	}
+    }
+
+    return 0;
+
+}
+
+/* **************************************************************** */
+/* **** make an entry in the les (2d) ***************************** */
+/* **************************************************************** */
+int make_les_entry_2d(int i, int j, int offset_i, int offset_j, int count,
+		      int pos, N_les * les, G_math_spvector * spvect,
+		      N_array_2d * cell_count, N_array_2d * status,
+		      N_array_2d * start_val, double entry, int cell_type)
+{
+    int K;
+    int di = offset_i;
+    int dj = offset_j;
+
+    K = N_get_array_2d_c_value(cell_count, i + di, j + dj) -
+	N_get_array_2d_c_value(cell_count, i, j);
+
+    /* active cells build the linear equation system */
+    if (cell_type == N_CELL_ACTIVE) {
+	/* dirichlet or transmission cells must be handled like this */
+	if (N_get_array_2d_c_value(status, i + di, j + dj) > N_CELL_ACTIVE &&
+	    N_get_array_2d_c_value(status, i + di, j + dj) < N_MAX_CELL_STATE)
+	    les->b[count] -=
+		N_get_array_2d_d_value(start_val, i + di, j + dj) * entry;
+	else if (N_get_array_2d_c_value(status, i + di, j + dj) ==
+		 N_CELL_ACTIVE) {
+	    if ((count + K) >= 0 && (count + K) < les->cols) {
+		G_debug(5,
+			" make_les_entry_2d: (N_CELL_ACTIVE) create matrix entry at row[%i] col[%i] value %g\n",
+			count, count + K, entry);
+		pos++;
+		if (les->type == N_SPARSE_LES) {
+		    spvect->index[pos] = count + K;
+		    spvect->values[pos] = entry;
+		}
+		else {
+		    les->A[count][count + K] = entry;
+		}
+	    }
+	}
+    }				/* if dirichlet cells should be used then check for all valid cell neighbours */
+    else if (cell_type == N_CELL_DIRICHLET) {
+	/* all valid cells */
+	if (N_get_array_2d_c_value(status, i + di, j + dj) > N_CELL_INACTIVE
+	    && N_get_array_2d_c_value(status, i + di,
+				      j + dj) < N_MAX_CELL_STATE) {
+	    if ((count + K) >= 0 && (count + K) < les->cols) {
+		G_debug(5,
+			" make_les_entry_2d: (N_CELL_DIRICHLET) create matrix entry at row[%i] col[%i] value %g\n",
+			count, count + K, entry);
+		pos++;
+		if (les->type == N_SPARSE_LES) {
+		    spvect->index[pos] = count + K;
+		    spvect->values[pos] = entry;
+		}
+		else {
+		    les->A[count][count + K] = entry;
+		}
+	    }
+	}
+    }
+
+    return pos;
+}
+
+
+/* *************************************************************** * 
+ * ******************** N_assemble_les_3d ************************ * 
+ * *************************************************************** */
+/*!
+ * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active cells
+ *
+ * This function calls #N_assemble_les_3d_param
+ * */
+N_les *N_assemble_les_3d(int les_type, N_geom_data * geom,
+			 N_array_3d * status, N_array_3d * start_val,
+			 void *data, N_les_callback_3d * call)
+{
+    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_ACTIVE);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active cells
+ *
+ * This function calls #N_assemble_les_3d_param
+ * */
+N_les *N_assemble_les_3d_active(int les_type, N_geom_data * geom,
+				N_array_3d * status, N_array_3d * start_val,
+				void *data, N_les_callback_3d * call)
+{
+    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_ACTIVE);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 3d location data (g3d) active and dirichlet cells
+ *
+ * This function calls #N_assemble_les_3d_param
+ * */
+N_les *N_assemble_les_3d_dirichlet(int les_type, N_geom_data * geom,
+				   N_array_3d * status,
+				   N_array_3d * start_val, void *data,
+				   N_les_callback_3d * call)
+{
+    return N_assemble_les_3d_param(les_type, geom, status, start_val, data,
+				   call, N_CELL_DIRICHLET);
+}
+
+/*!
+ * \brief Assemble a linear equation system (les) based on 3d location data (g3d)
+ *
+ * The linear equation system type can be set to N_NORMAL_LES to create a regular
+ * matrix, or to N_SPARSE_LES to create a sparse matrix. This function returns
+ * a new created linear equation system which can be solved with 
+ * linear equation solvers. An 3d array with start values and an 3d status array
+ * must be provided as well as the location geometry and a void pointer to data 
+ * passed to the callback which creates the les row entries. This callback
+ * must be defined in the N_les_callback_3d structure.
+ * 
+ * The creation of the les is parallelized with OpenMP. 
+ * If you implement new callbacks, please make sure that the 
+ * function calls are thread safe.
+ *
+ * the les can be created in two ways, with dirichlet and similar cells and without them,
+ * to spare some memory. If the les is created with dirichlet cell, the dirichlet boundary condition
+ * must be added.
+ *
+ * \param les_type int
+ * \param geom      N_geom_data*
+ * \param status    N_array_3d *
+ * \param start_val N_array_3d *
+ * \param data void *
+ * \param call N_les_callback_3d *
+ * \param cell_type int  -- les assemble based on N_CELL_ACTIVE or N_CELL_DIRICHLET
+ * \return N_les *
+ * */
+N_les *N_assemble_les_3d_param(int les_type, N_geom_data * geom,
+			       N_array_3d * status, N_array_3d * start_val,
+			       void *data, N_les_callback_3d * call,
+			       int cell_type)
+{
+    int i, j, k, count = 0, pos = 0;
+    int cell_type_count = 0;
+    N_array_3d *cell_count;
+    N_les *les = NULL;
+    int **index_ij;
+
+    G_debug(2,
+	    "N_assemble_les_3d: starting to assemble the linear equation system");
+
+    cell_count =
+	N_alloc_array_3d(geom->cols, geom->rows, geom->depths, 1, DCELL_TYPE);
+
+    /* First count the number of valid cells and save  
+     * each number in a new 3d array. Those numbers are used 
+     * to create the linear equation system.*/
+
+    if (cell_type == N_CELL_DIRICHLET) {
+	/* include dirichlet cells in the les */
+	for (k = 0; k < geom->depths; k++) {
+	    for (j = 0; j < geom->rows; j++) {
+		for (i = 0; i < geom->cols; i++) {
+		    /*use all non-inactive cells for les creation */
+		    if (N_CELL_INACTIVE <
+			(int)N_get_array_3d_d_value(status, i, j, k) &&
+			(int)N_get_array_3d_d_value(status, i, j,
+						    k) < N_MAX_CELL_STATE)
+			cell_type_count++;
+		}
+	    }
+	}
+    }
+    else {
+	/*use only active cell in the les */
+	for (k = 0; k < geom->depths; k++) {
+	    for (j = 0; j < geom->rows; j++) {
+		for (i = 0; i < geom->cols; i++) {
+		    /*count only active cells */
+		    if (N_CELL_ACTIVE
+			== (int)N_get_array_3d_d_value(status, i, j, k))
+			cell_type_count++;
+
+		}
+	    }
+	}
+    }
+
+    G_debug(2,
+	    "N_assemble_les_3d: number of  used cells %i\n", cell_type_count);
+
+    if (cell_type_count == 0.0)
+	G_fatal_error
+	    ("Not enough active cells [%i] to create the linear equation system. Check the cell status. Only active cells (value = 1) are used to create the equation system.",
+	     cell_type_count);
+
+    /* allocate the memory for the linear equation system (les). 
+     * Only valid cells are used to create the les. */
+    les = N_alloc_les_Ax_b(cell_type_count, les_type);
+
+    index_ij = (int **)G_calloc(cell_type_count, sizeof(int *));
+    for (i = 0; i < cell_type_count; i++)
+	index_ij[i] = (int *)G_calloc(3, sizeof(int));
+
+    count = 0;
+    /*count the number of cells which should be used to create the linear equation system */
+    /*save the k, i and j indices and create a ordered numbering */
+    for (k = 0; k < geom->depths; k++) {
+	for (j = 0; j < geom->rows; j++) {
+	    for (i = 0; i < geom->cols; i++) {
+		if (cell_type == N_CELL_DIRICHLET) {
+		    if (N_CELL_INACTIVE <
+			(int)N_get_array_3d_d_value(status, i, j, k) &&
+			(int)N_get_array_3d_d_value(status, i, j,
+						    k) < N_MAX_CELL_STATE) {
+			N_put_array_3d_d_value(cell_count, i, j, k, count);
+			index_ij[count][0] = i;
+			index_ij[count][1] = j;
+			index_ij[count][2] = k;
+			count++;
+			G_debug(5,
+				"N_assemble_les_3d: non-inactive cells count %i at pos x[%i] y[%i] z[%i]\n",
+				count, i, j, k);
+		    }
+		}
+		else if (N_CELL_ACTIVE ==
+			 (int)N_get_array_3d_d_value(status, i, j, k)) {
+		    N_put_array_3d_d_value(cell_count, i, j, k, count);
+		    index_ij[count][0] = i;
+		    index_ij[count][1] = j;
+		    index_ij[count][2] = k;
+		    count++;
+		    G_debug(5,
+			    "N_assemble_les_3d: active cells count %i at pos x[%i] y[%i] z[%i]\n",
+			    count, i, j, k);
+		}
+	    }
+	}
+    }
+
+    G_debug(2, "N_assemble_les_3d: starting the parallel assemble loop");
+
+#pragma omp parallel for private(i, j, k, pos, count) schedule(static)
+    for (count = 0; count < cell_type_count; count++) {
+	i = index_ij[count][0];
+	j = index_ij[count][1];
+	k = index_ij[count][2];
+
+	/*create the entries for the */
+	N_data_star *items = call->callback(data, geom, i, j, k);
+
+	G_math_spvector *spvect = NULL;
+
+	/*allocate a sprase vector */
+	if (les_type == N_SPARSE_LES)
+	    spvect = G_math_alloc_spvector(items->count);
+	/* initial conditions */
+
+	les->x[count] = N_get_array_3d_d_value(start_val, i, j, k);
+
+	/* the entry in the vector b */
+	les->b[count] = items->V;
+
+	/* pos describes the position in the sparse vector.
+	 * the first entry is always the diagonal entry of the matrix*/
+	pos = 0;
+
+	if (les_type == N_SPARSE_LES) {
+	    spvect->index[pos] = count;
+	    spvect->values[pos] = items->C;
+	}
+	else {
+	    les->A[count][count] = items->C;
+	}
+	/* western neighbour, entry is col - 1 */
+	if (i > 0) {
+	    pos =
+		make_les_entry_3d(i, j, k, -1, 0, 0, count, pos, les, spvect,
+				  cell_count, status, start_val, items->W,
+				  cell_type);
+	}
+	/* eastern neighbour, entry col + 1 */
+	if (i < geom->cols - 1) {
+	    pos = make_les_entry_3d(i, j, k, 1, 0, 0, count, pos, les, spvect,
+				    cell_count, status, start_val, items->E,
+				    cell_type);
+	}
+	/* northern neighbour, entry row -1 */
+	if (j > 0) {
+	    pos =
+		make_les_entry_3d(i, j, k, 0, -1, 0, count, pos, les, spvect,
+				  cell_count, status, start_val, items->N,
+				  cell_type);
+	}
+	/* southern neighbour, entry row +1 */
+	if (j < geom->rows - 1) {
+	    pos = make_les_entry_3d(i, j, k, 0, 1, 0, count, pos, les, spvect,
+				    cell_count, status, start_val, items->S,
+				    cell_type);
+	}
+	/*only for a 7 star entry needed */
+	if (items->type == N_7_POINT_STAR || items->type == N_27_POINT_STAR) {
+	    /* the upper cell (top), entry depth + 1 */
+	    if (k < geom->depths - 1) {
+		pos =
+		    make_les_entry_3d(i, j, k, 0, 0, 1, count, pos, les,
+				      spvect, cell_count, status, start_val,
+				      items->T, cell_type);
+	    }
+	    /* the lower cell (bottom), entry depth - 1 */
+	    if (k > 0) {
+		pos =
+		    make_les_entry_3d(i, j, k, 0, 0, -1, count, pos, les,
+				      spvect, cell_count, status, start_val,
+				      items->B, cell_type);
+	    }
+	}
+
+	/*How many entries in the les */
+	if (les->type == N_SPARSE_LES) {
+	    spvect->cols = pos + 1;
+	    G_math_add_spvector(les->Asp, spvect, count);
+	}
+
+	if (items)
+	    G_free(items);
+    }
+
+    N_free_array_3d(cell_count);
+
+    for (i = 0; i < cell_type_count; i++)
+	G_free(index_ij[i]);
+
+    G_free(index_ij);
+
+    return les;
+}
+
+/*!
+ * \brief Integrate Dirichlet or Transmission boundary conditions into the les (3d)
+ *
+ * Dirichlet and Transmission boundary conditions will be integrated into
+ * the provided linear equation system. This is meaningfull if
+ * the les was created with #N_assemble_les_2d_dirichlet, because in
+ * this case Dirichlet boundary conditions are not automatically included.
+ *
+ * The provided les will be modified:
+ *
+ * Ax = b will be splitted into Ax_u + Ax_d = b
+ *
+ * x_u - the unknowns
+ * x_d - the Dirichlet cells
+ *
+ * Ax_u = b -Ax_d will be computed. Then the matrix A will be modified to
+ *
+ * | A_u  0 | x_u
+ * |  0   I | x_d
+ *
+ * \param les N_les* -- the linear equation system
+ * \param geom N_geom_data* -- geometrical data information
+ * \param status N_array_2d* -- the status array containing the cell types
+ * \param start_val N_array_2d* -- an array with start values
+ * \return int -- 1 = success, 0 = failure
+ * */
+int N_les_integrate_dirichlet_3d(N_les * les, N_geom_data * geom,
+				 N_array_3d * status, N_array_3d * start_val)
+{
+    int rows, cols, depths;
+    int count = 0;
+    int i, j, x, y, z, stat;
+    double *dvect1;
+    double *dvect2;
+
+    G_debug(2,
+	    "N_les_integrate_dirichlet_3d: integrating the dirichlet boundary condition");
+
+    rows = geom->rows;
+    cols = geom->cols;
+    depths = geom->depths;
+
+    /*we nned to additional vectors */
+    dvect1 = (double *)G_calloc(les->cols, sizeof(double));
+    dvect2 = (double *)G_calloc(les->cols, sizeof(double));
+
+    /*fill the first one with the x vector data of Dirichlet cells */
+    count = 0;
+    for (z = 0; z < depths; z++) {
+	for (y = 0; y < rows; y++) {
+	    for (x = 0; x < cols; x++) {
+		stat = (int)N_get_array_3d_d_value(status, x, y, z);
+		if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
+		    dvect1[count] =
+			N_get_array_3d_d_value(start_val, x, y, z);
+		    count++;
+		}
+		else if (stat == N_CELL_ACTIVE) {
+		    dvect1[count] = 0.0;
+		    count++;
+		}
+	    }
+	}
+    }
+
+#pragma omp parallel default(shared)
+    {
+	/*perform the matrix vector product and */
+	if (les->type == N_SPARSE_LES)
+	    G_math_Ax_sparse(les->Asp, dvect1, dvect2, les->rows);
+	else
+	    G_math_d_Ax(les->A, dvect1, dvect2, les->rows, les->cols);
+#pragma omp for schedule (static) private(i)
+	for (i = 0; i < les->cols; i++)
+	    les->b[i] = les->b[i] - dvect2[i];
+    }
+
+    /*now set the Dirichlet cell rows and cols to zero and the 
+     * diagonal entry to 1*/
+    count = 0;
+    for (z = 0; z < depths; z++) {
+	for (y = 0; y < rows; y++) {
+	    for (x = 0; x < cols; x++) {
+		stat = (int)N_get_array_3d_d_value(status, x, y, z);
+		if (stat > N_CELL_ACTIVE && stat < N_MAX_CELL_STATE) {
+		    if (les->type == N_SPARSE_LES) {
+			/*set the rows to zero */
+			for (i = 0; i < les->Asp[count]->cols; i++)
+			    les->Asp[count]->values[i] = 0.0;
+			/*set the cols to zero */
+			for (i = 0; i < les->rows; i++) {
+			    for (j = 0; j < les->Asp[i]->cols; j++) {
+				if (les->Asp[i]->index[j] == count)
+				    les->Asp[i]->values[j] = 0.0;
+			    }
+			}
+
+			/*entry on the diagonal */
+			les->Asp[count]->values[0] = 1.0;
+
+		    }
+		    else {
+			/*set the rows to zero */
+			for (i = 0; i < les->cols; i++)
+			    les->A[count][i] = 0.0;
+			/*set the cols to zero */
+			for (i = 0; i < les->rows; i++)
+			    les->A[i][count] = 0.0;
+
+			/*entry on the diagonal */
+			les->A[count][count] = 1.0;
+		    }
+		}
+		count++;
+	    }
+	}
+    }
+
+    return 0;
+
+}
+
+/* **************************************************************** */
+/* **** make an entry in the les (3d) ***************************** */
+/* **************************************************************** */
+int make_les_entry_3d(int i, int j, int k, int offset_i, int offset_j,
+		      int offset_k, int count, int pos, N_les * les,
+		      G_math_spvector * spvect, N_array_3d * cell_count,
+		      N_array_3d * status, N_array_3d * start_val,
+		      double entry, int cell_type)
+{
+    int K;
+    int di = offset_i;
+    int dj = offset_j;
+    int dk = offset_k;
+
+    K = (int)N_get_array_3d_d_value(cell_count, i + di, j + dj, k + dk) -
+	(int)N_get_array_3d_d_value(cell_count, i, j, k);
+
+    if (cell_type == N_CELL_ACTIVE) {
+	if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk) >
+	    N_CELL_ACTIVE &&
+	    (int)N_get_array_3d_d_value(status, i + di, j + dj,
+					k + dk) < N_MAX_CELL_STATE)
+	    les->b[count] -=
+		N_get_array_3d_d_value(start_val, i + di, j + dj,
+				       k + dk) * entry;
+	else if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk)
+		 == N_CELL_ACTIVE) {
+	    if ((count + K) >= 0 && (count + K) < les->cols) {
+		G_debug(5,
+			" make_les_entry_3d: (N_CELL_ACTIVE) create matrix entry at row[%i] col[%i] value %g\n",
+			count, count + K, entry);
+		pos++;
+		if (les->type == N_SPARSE_LES) {
+		    spvect->index[pos] = count + K;
+		    spvect->values[pos] = entry;
+		}
+		else {
+		    les->A[count][count + K] = entry;
+		}
+	    }
+	}
+    }
+    else if (cell_type == N_CELL_DIRICHLET) {
+	if ((int)N_get_array_3d_d_value(status, i + di, j + dj, k + dk)
+	    != N_CELL_INACTIVE) {
+	    if ((count + K) >= 0 && (count + K) < les->cols) {
+		G_debug(5,
+			" make_les_entry_3d: (N_CELL_DIRICHLET) create matrix entry at row[%i] col[%i] value %g\n",
+			count, count + K, entry);
+		pos++;
+		if (les->type == N_SPARSE_LES) {
+		    spvect->index[pos] = count + K;
+		    spvect->values[pos] = entry;
+		}
+		else {
+		    les->A[count][count + K] = entry;
+		}
+	    }
+	}
+    }
+
+    return pos;
+}

Copied: grass/trunk/lib/gpde/n_parse_options.c (from rev 62429, grass/trunk/lib/gpde/N_parse_options.c)
===================================================================
--- grass/trunk/lib/gpde/n_parse_options.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_parse_options.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,111 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      standard parser option for the numerical pde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <grass/glocale.h>
+#include <grass/N_pde.h>
+
+/*!
+ * \brief Create standardised Option structure related to the gpde library.
+ *
+ * This function will create a standardised Option structure
+ * defined by parameter opt. A list of valid parameters can be found in N_pde.h.
+ * It allocates memory for the Option structure and returns a pointer to
+ * this memory (of <i>type struct Option *</i>).<br>
+ *
+ * If an invalid parameter was specified an empty Option structure will
+ * be returned (not NULL).
+ *
+ * This function is related to the gpde library, general standard options can be 
+ * found in lib/gis/parser.c. These options are set with G_define_standard_option ();
+ *
+ * \param[in] opt Type of Option struct to create
+ * \return Option * Pointer to an Option struct
+ *
+ * */
+struct Option *N_define_standard_option(int opt)
+{
+    struct Option *Opt;
+
+    Opt = G_define_option();
+
+    switch (opt) {
+	/*solver for symmetric, positive definite linear equation systems */
+    case N_OPT_SOLVER_SYMM:
+	Opt->key = "solver";
+	Opt->type = TYPE_STRING;
+	Opt->required = NO;
+	Opt->key_desc = "name";
+	Opt->answer = "cg";
+	Opt->options = "gauss,lu,cholesky,jacobi,sor,cg,bicgstab,pcg";
+        Opt->guisection = "Solver";
+	Opt->description =
+	    ("The type of solver which should solve the symmetric linear equation system");
+	break;
+	/*solver for unsymmetric linear equation systems */
+    case N_OPT_SOLVER_UNSYMM:
+	Opt->key = "solver";
+	Opt->type = TYPE_STRING;
+	Opt->required = NO;
+	Opt->key_desc = "name";
+	Opt->answer = "bicgstab";
+	Opt->options = "gauss,lu,jacobi,sor,bicgstab";
+        Opt->guisection = "Solver";
+	Opt->description =
+	    ("The type of solver which should solve the linear equation system");
+	break;
+    case N_OPT_MAX_ITERATIONS:
+	Opt->key = "maxit";
+	Opt->type = TYPE_INTEGER;
+	Opt->required = NO;
+	Opt->answer = "10000";
+        Opt->guisection = "Solver";
+	Opt->description =
+	    ("Maximum number of iteration used to solve the linear equation system");
+	break;
+    case N_OPT_ITERATION_ERROR:
+	Opt->key = "error";
+	Opt->type = TYPE_DOUBLE;
+	Opt->required = NO;
+	Opt->answer = "0.000001";
+        Opt->guisection = "Solver";
+	Opt->description =
+	    ("Error break criteria for iterative solver");
+	break;
+    case N_OPT_SOR_VALUE:
+	Opt->key = "relax";
+	Opt->type = TYPE_DOUBLE;
+	Opt->required = NO;
+	Opt->answer = "1";
+        Opt->guisection = "Solver";
+	Opt->description =
+	    ("The relaxation parameter used by the jacobi and sor solver for speedup or stabilizing");
+	break;
+    case N_OPT_CALC_TIME:
+	Opt->key = "dt";
+	Opt->type = TYPE_DOUBLE;
+	Opt->required = YES;
+	Opt->answer = "86400";
+        Opt->guisection = "Solver";
+	Opt->description = _("The calculation time in seconds");
+	break;
+    }
+
+    return Opt;
+}

Copied: grass/trunk/lib/gpde/n_solute_transport.c (from rev 62429, grass/trunk/lib/gpde/N_solute_transport.c)
===================================================================
--- grass/trunk/lib/gpde/n_solute_transport.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_solute_transport.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,772 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      solute transport in porous media
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2007 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <math.h>
+#include <grass/N_solute_transport.h>
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*! \brief This is just a placeholder
+ *
+ * */
+N_data_star *N_callback_solute_transport_3d(void *solutedata,
+					    N_geom_data * geom, int col,
+					    int row, int depth)
+{
+    double Df_e = 0, Df_w = 0, Df_n = 0, Df_s = 0, Df_t = 0, Df_b = 0;
+    double dx, dy, dz, Az;
+    double diff_x, diff_y, diff_z;
+    double diff_xw, diff_yn;
+    double diff_xe, diff_ys;
+    double diff_zt, diff_zb;
+    double cin = 0, cg, cg_start;
+    double R, nf, cs, q;
+    double C, W, E, N, S, T, B, V;
+    double vw = 0, ve = 0, vn = 0, vs = 0, vt = 0, vb = 0;
+    double Ds_w = 0, Ds_e = 0, Ds_n = 0, Ds_s = 0, Ds_t = 0, Ds_b = 0;
+    double Dw = 0, De = 0, Dn = 0, Ds = 0, Dt = 0, Db = 0;
+    double rw = 0.5, re = 0.5, rn = 0.5, rs = 0.5, rt = 0.5, rb = 0.5;
+
+    N_solute_transport_data3d *data = NULL;
+    N_data_star *mat_pos;
+    N_gradient_3d grad;
+
+    /*cast the void pointer to the right data structure */
+    data = (N_solute_transport_data3d *) solutedata;
+
+    N_get_gradient_3d(data->grad, &grad, col, row, depth);
+
+    dx = geom->dx;
+    dy = geom->dy;
+    dz = geom->dz;
+    Az = N_get_geom_data_area_of_cell(geom, row);
+
+    /*read the data from the arrays */
+    cg_start = N_get_array_3d_d_value(data->c_start, col, row, depth);
+    cg = N_get_array_3d_d_value(data->c, col, row, depth);
+
+    /*get the surrounding diffusion tensor entries */
+    diff_x = N_get_array_3d_d_value(data->diff_x, col, row, depth);
+    diff_y = N_get_array_3d_d_value(data->diff_y, col, row, depth);
+    diff_z = N_get_array_3d_d_value(data->diff_z, col, row, depth);
+    diff_xw = N_get_array_3d_d_value(data->diff_x, col - 1, row, depth);
+    diff_xe = N_get_array_3d_d_value(data->diff_x, col + 1, row, depth);
+    diff_yn = N_get_array_3d_d_value(data->diff_y, col, row - 1, depth);
+    diff_ys = N_get_array_3d_d_value(data->diff_y, col, row + 1, depth);
+    diff_zt = N_get_array_3d_d_value(data->diff_z, col, row, depth + 1);
+    diff_zb = N_get_array_3d_d_value(data->diff_z, col, row, depth - 1);
+
+    /* calculate the diffusion on the cell borders using the harmonical mean */
+    Df_w = N_calc_harmonic_mean(diff_xw, diff_x);
+    Df_e = N_calc_harmonic_mean(diff_xe, diff_x);
+    Df_n = N_calc_harmonic_mean(diff_yn, diff_y);
+    Df_s = N_calc_harmonic_mean(diff_ys, diff_y);
+    Df_t = N_calc_harmonic_mean(diff_zt, diff_z);
+    Df_b = N_calc_harmonic_mean(diff_zb, diff_z);
+
+    /* calculate the dispersion */
+    /*todo */
+
+    /* calculate the velocity parts  with full upwinding scheme */
+    vw = grad.WC;
+    ve = grad.EC;
+    vn = grad.NC;
+    vs = grad.SC;
+    vt = grad.TC;
+    vb = grad.BC;
+
+    /* put the diffusion and dispersion together */
+    Dw = ((Df_w + Ds_w)) / dx;
+    De = ((Df_e + Ds_e)) / dx;
+    Dn = ((Df_n + Ds_n)) / dy;
+    Ds = ((Df_s + Ds_s)) / dy;
+    Dt = ((Df_t + Ds_t)) / dz;
+    Db = ((Df_b + Ds_b)) / dz;
+
+    rw = N_exp_upwinding(-1 * vw, dx, Dw);
+    re = N_exp_upwinding(ve, dx, De);
+    rs = N_exp_upwinding(-1 * vs, dy, Ds);
+    rn = N_exp_upwinding(vn, dy, Dn);
+    rb = N_exp_upwinding(-1 * vb, dz, Dn);
+    rt = N_exp_upwinding(vt, dz, Dn);
+
+    /*mass balance center cell to western cell */
+    W = -1 * (Dw) * dy * dz - vw * (1 - rw) * dy * dz;
+    /*mass balance center cell to eastern cell */
+    E = -1 * (De) * dy * dz + ve * (1 - re) * dy * dz;
+    /*mass balance center cell to southern cell */
+    S = -1 * (Ds) * dx * dz - vs * (1 - rs) * dx * dz;
+    /*mass balance center cell to northern cell */
+    N = -1 * (Dn) * dx * dz + vn * (1 - rn) * dx * dz;
+    /*mass balance center cell to bottom cell */
+    B = -1 * (Db) * Az - vb * (1 - rb) * Az;
+    /*mass balance center cell to top cell */
+    T = -1 * (Dt) * Az + vt * (1 - rt) * Az;
+
+    /* Retardation */
+    R = N_get_array_3d_d_value(data->R, col, row, depth);
+    /* Inner sources */
+    cs = N_get_array_3d_d_value(data->cs, col, row, depth);
+    /* effective porosity */
+    nf = N_get_array_3d_d_value(data->nf, col, row, depth);
+    /* groundwater sources and sinks */
+    q = N_get_array_3d_d_value(data->q, col, row, depth);
+    /* concentration of influent water */
+    cin = N_get_array_3d_d_value(data->cin, col, row, depth);
+
+    /*the diagonal entry of the matrix */
+    C = ((Dw - vw) * dy * dz +
+	 (De + ve) * dy * dz +
+	 (Ds - vs) * dx * dz +
+	 (Dn + vn) * dx * dz +
+	 (Db - vb) * Az + (Dt + vt) * Az + Az * dz * R / data->dt - q / nf);
+
+    /*the entry in the right side b of Ax = b */
+    V = (cs + cg_start * Az * dz * R / data->dt - q / nf * cin);
+
+    /*
+     * printf("nf %g\n", nf);
+     * printf("q %g\n", q);
+     * printf("cs %g\n", cs);
+     * printf("cin %g\n", cin);
+     * printf("cg %g\n", cg);
+     * printf("cg_start %g\n", cg_start);
+     * printf("Az %g\n", Az);
+     * printf("z %g\n", z);
+     * printf("R %g\n", R);
+     * printf("dt %g\n", data->dt);
+     */
+    G_debug(6, "N_callback_solute_transport_3d: called [%i][%i][%i]", row,
+	    col, depth);
+
+    /*create the 7 point star entries */
+    mat_pos = N_create_7star(C, W, E, N, S, T, B, V);
+
+    return mat_pos;
+}
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*!
+ * \brief This callback function creates the mass balance of a 5 point star
+ *
+ * The mass balance is based on the common solute transport equation:
+ *
+ * \f[\frac{\partial c_g}{\partial t} R = \nabla \cdot ({\bf D} \nabla c_g - {\bf u} c_g) + \sigma + \frac{q}{n_f}(c_g - c_in) \f]
+ *
+ * This equation is discretizised with the finite volume method in two dimensions.
+ *
+ *
+ * \param solutedata  * N_solute_transport_data2d - a void pointer to the data structure
+ * \param geom N_geom_data *
+ * \param col   int
+ * \param row   int
+ * \return N_data_star * - a five point data star
+ *
+ * */
+N_data_star *N_callback_solute_transport_2d(void *solutedata,
+					    N_geom_data * geom, int col,
+					    int row)
+{
+    double Df_e = 0, Df_w = 0, Df_n = 0, Df_s = 0;
+    double z_e = 0, z_w = 0, z_n = 0, z_s = 0;
+    double dx, dy, Az;
+    double diff_x, diff_y;
+    double disp_x, disp_y;
+    double z;
+    double diff_xw, diff_yn;
+    double disp_xw, disp_yn;
+    double z_xw, z_yn;
+    double diff_xe, diff_ys;
+    double disp_xe, disp_ys;
+    double z_xe, z_ys;
+    double cin = 0, cg, cg_start;
+    double R, nf, cs, q;
+    double C, W, E, N, S, V, NE, NW, SW, SE;
+    double vw = 0, ve = 0, vn = 0, vs = 0;
+    double Ds_w = 0, Ds_e = 0, Ds_n = 0, Ds_s = 0;
+    double Dw = 0, De = 0, Dn = 0, Ds = 0;
+    double rw = 0.5, re = 0.5, rn = 0.5, rs = 0.5;
+
+    N_solute_transport_data2d *data = NULL;
+    N_data_star *mat_pos;
+    N_gradient_2d grad;
+
+    /*cast the void pointer to the right data structure */
+    data = (N_solute_transport_data2d *) solutedata;
+
+    N_get_gradient_2d(data->grad, &grad, col, row);
+
+    dx = geom->dx;
+    dy = geom->dy;
+    Az = N_get_geom_data_area_of_cell(geom, row);
+
+    /*read the data from the arrays */
+    cg_start = N_get_array_2d_d_value(data->c_start, col, row);
+    cg = N_get_array_2d_d_value(data->c, col, row);
+
+    /* calculate the cell height */
+    z = N_get_array_2d_d_value(data->top, col,
+			       row) -
+	N_get_array_2d_d_value(data->bottom, col, row);
+    z_xw =
+	N_get_array_2d_d_value(data->top, col - 1,
+			       row) -
+	N_get_array_2d_d_value(data->bottom, col - 1, row);
+    z_xe =
+	N_get_array_2d_d_value(data->top, col + 1,
+			       row) -
+	N_get_array_2d_d_value(data->bottom, col + 1, row);
+    z_yn =
+	N_get_array_2d_d_value(data->top, col,
+			       row - 1) -
+	N_get_array_2d_d_value(data->bottom, col, row - 1);
+    z_ys =
+	N_get_array_2d_d_value(data->top, col,
+			       row + 1) -
+	N_get_array_2d_d_value(data->bottom, col, row + 1);
+
+    /*geometrical mean of cell height */
+    z_w = N_calc_geom_mean(z_xw, z);
+    z_e = N_calc_geom_mean(z_xe, z);
+    z_n = N_calc_geom_mean(z_yn, z);
+    z_s = N_calc_geom_mean(z_ys, z);
+
+    /*get the surrounding diffusion tensor entries */
+    diff_x = N_get_array_2d_d_value(data->diff_x, col, row);
+    diff_y = N_get_array_2d_d_value(data->diff_y, col, row);
+    diff_xw = N_get_array_2d_d_value(data->diff_x, col - 1, row);
+    diff_xe = N_get_array_2d_d_value(data->diff_x, col + 1, row);
+    diff_yn = N_get_array_2d_d_value(data->diff_y, col, row - 1);
+    diff_ys = N_get_array_2d_d_value(data->diff_y, col, row + 1);
+
+    /* calculate the diffusion at the cell borders using the harmonical mean */
+    Df_w = N_calc_harmonic_mean(diff_xw, diff_x);
+    Df_e = N_calc_harmonic_mean(diff_xe, diff_x);
+    Df_n = N_calc_harmonic_mean(diff_yn, diff_y);
+    Df_s = N_calc_harmonic_mean(diff_ys, diff_y);
+
+    /* calculate the dispersion */
+    /*get the surrounding dispersion tensor entries */
+    disp_x = N_get_array_2d_d_value(data->disp_xx, col, row);
+    disp_y = N_get_array_2d_d_value(data->disp_yy, col, row);
+    if (N_get_array_2d_d_value(data->status, col - 1, row) ==
+	N_CELL_TRANSMISSION) {
+	disp_xw = disp_x;
+    }
+    else {
+	disp_xw = N_get_array_2d_d_value(data->disp_xx, col - 1, row);
+    }
+    if (N_get_array_2d_d_value(data->status, col + 1, row) ==
+	N_CELL_TRANSMISSION) {
+	disp_xe = disp_x;
+    }
+    else {
+	disp_xe = N_get_array_2d_d_value(data->disp_xx, col + 1, row);
+    }
+    if (N_get_array_2d_d_value(data->status, col, row - 1) ==
+	N_CELL_TRANSMISSION) {
+	disp_yn = disp_y;
+    }
+    else {
+	disp_yn = N_get_array_2d_d_value(data->disp_yy, col, row - 1);
+    }
+    if (N_get_array_2d_d_value(data->status, col, row + 1) ==
+	N_CELL_TRANSMISSION) {
+	disp_ys = disp_y;
+    }
+    else {
+	disp_ys = N_get_array_2d_d_value(data->disp_yy, col, row + 1);
+    }
+
+    /* calculate the dispersion at the cell borders using the harmonical mean */
+    Ds_w = N_calc_harmonic_mean(disp_xw, disp_x);
+    Ds_e = N_calc_harmonic_mean(disp_xe, disp_x);
+    Ds_n = N_calc_harmonic_mean(disp_yn, disp_y);
+    Ds_s = N_calc_harmonic_mean(disp_ys, disp_y);
+
+    /* put the diffusion and dispersion together */
+    Dw = ((Df_w + Ds_w)) / dx;
+    De = ((Df_e + Ds_e)) / dx;
+    Ds = ((Df_s + Ds_s)) / dy;
+    Dn = ((Df_n + Ds_n)) / dy;
+
+    vw = -1.0 * grad.WC;
+    ve = grad.EC;
+    vs = -1.0 * grad.SC;
+    vn = grad.NC;
+
+    if (data->stab == N_UPWIND_FULL) {
+	rw = N_full_upwinding(vw, dx, Dw);
+	re = N_full_upwinding(ve, dx, De);
+	rs = N_full_upwinding(vs, dy, Ds);
+	rn = N_full_upwinding(vn, dy, Dn);
+    }
+    else if (data->stab == N_UPWIND_EXP) {
+	rw = N_exp_upwinding(vw, dx, Dw);
+	re = N_exp_upwinding(ve, dx, De);
+	rs = N_exp_upwinding(vs, dy, Ds);
+	rn = N_exp_upwinding(vn, dy, Dn);
+    }
+
+    /*mass balance center cell to western cell */
+    W = -1 * (Dw) * dy * z_w + vw * (1 - rw) * dy * z_w;
+    /*mass balance center cell to eastern cell */
+    E = -1 * (De) * dy * z_e + ve * (1 - re) * dy * z_e;
+    /*mass balance center cell to southern cell */
+    S = -1 * (Ds) * dx * z_s + vs * (1 - rs) * dx * z_s;
+    /*mass balance center cell to northern cell */
+    N = -1 * (Dn) * dx * z_n + vn * (1 - rn) * dx * z_n;
+
+    NW = 0.0;
+    SW = 0.0;
+    NE = 0.0;
+    SE = 0.0;
+
+    /* Retardation */
+    R = N_get_array_2d_d_value(data->R, col, row);
+    /* Inner sources */
+    cs = N_get_array_2d_d_value(data->cs, col, row);
+    /* effective porosity */
+    nf = N_get_array_2d_d_value(data->nf, col, row);
+    /* groundwater sources and sinks */
+    q = N_get_array_2d_d_value(data->q, col, row);
+    /* concentration of influent water */
+    cin = N_get_array_2d_d_value(data->cin, col, row);
+
+    /*the diagonal entry of the matrix */
+    C = (Dw + vw * rw) * dy * z_w +
+	(De + ve * re) * dy * z_e +
+	(Ds + vs * rs) * dx * z_s +
+	(Dn + vn * rn) * dx * z_n + Az * z * R / data->dt - q / nf;
+
+    /*the entry in the right side b of Ax = b */
+    V = (cs + cg_start * Az * z * R / data->dt + q / nf * cin);
+
+    /*
+       fprintf(stderr, "nf %g\n", nf);
+       fprintf(stderr, "q %g\n", q);
+       fprintf(stderr, "cs %g\n", cs);
+       fprintf(stderr, "cin %g\n", cin);
+       fprintf(stderr, "cg %g\n", cg);
+       fprintf(stderr, "cg_start %g\n", cg_start);
+       fprintf(stderr, "Az %g\n", Az);
+       fprintf(stderr, "z %g\n", z);
+       fprintf(stderr, "R %g\n", R);
+       fprintf(stderr, "dt %g\n", data->dt);
+     */
+
+    G_debug(6, "N_callback_solute_transport_2d: called [%i][%i]", row, col);
+
+    /*create the 9 point star entries */
+    mat_pos = N_create_9star(C, W, E, N, S, NW, SW, NE, SE, V);
+
+    return mat_pos;
+}
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*!
+ * \brief Alllocate memory for the solute transport data structure in three dimensions
+ *
+ * The solute transport data structure will be allocated including
+ * all appendant 3d arrays. The offset for the 3d arrays is one
+ * to establish homogeneous Neumann boundary conditions at the calculation area border.
+ * This data structure is used to create a linear equation system based on the computation of
+ * solute transport in porous media with the finite volume method.
+ *
+ * \param cols   int
+ * \param rows   int
+ * \param depths int
+ * \return N_solute_transport_data3d *
+ * */
+
+N_solute_transport_data3d *N_alloc_solute_transport_data3d(int cols, int rows,
+							   int depths)
+{
+    N_solute_transport_data3d *data = NULL;
+
+    data =
+	(N_solute_transport_data3d *) G_calloc(1,
+					       sizeof
+					       (N_solute_transport_data3d));
+
+    data->c = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->c_start = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->status = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->diff_x = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->diff_y = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->diff_z = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->q = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->cs = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->R = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->nf = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->cin = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+
+    /*Allocate the dispersivity tensor */
+    data->disp_xx = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->disp_yy = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->disp_zz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->disp_xy = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->disp_xz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+    data->disp_yz = N_alloc_array_3d(cols, rows, depths, 1, DCELL_TYPE);
+
+
+    data->grad = N_alloc_gradient_field_3d(cols, rows, depths);
+    data->stab = N_UPWIND_EXP;
+
+    return data;
+}
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*!
+ * \brief Alllocate memory for the solute transport data structure in two dimensions
+ *
+ * The solute transport data structure will be allocated including
+ * all appendant 2d arrays. The offset for the 2d arrays is one
+ * to establish homogeneous Neumann boundary conditions at the calculation area border.
+ * This data structure is used to create a linear equation system based on the computation of
+ * solute transport in porous media with the finite volume method.
+ *
+ * \param cols   int
+ * \param rows   int
+ * \return N_solute_transport_data2d *
+ * */
+
+
+N_solute_transport_data2d *N_alloc_solute_transport_data2d(int cols, int rows)
+{
+    N_solute_transport_data2d *data = NULL;
+
+    data =
+	(N_solute_transport_data2d *) G_calloc(1,
+					       sizeof
+					       (N_solute_transport_data2d));
+
+    data->c = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->c_start = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->status = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->diff_x = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->diff_y = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->q = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->cs = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->R = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->nf = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->cin = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->top = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->bottom = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+
+    /*Allocate the dispersivity tensor */
+    data->disp_xx = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->disp_yy = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+    data->disp_xy = N_alloc_array_2d(cols, rows, 1, DCELL_TYPE);
+
+    data->grad = N_alloc_gradient_field_2d(cols, rows);
+    data->stab = N_UPWIND_EXP;
+
+    return data;
+}
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*!
+ * \brief Release the memory of the solute transport data structure in three dimensions
+ *
+ * \param data N_solute_transport_data2d *
+ * \return void *
+ * */
+void N_free_solute_transport_data3d(N_solute_transport_data3d * data)
+{
+    N_free_array_3d(data->c);
+    N_free_array_3d(data->c_start);
+    N_free_array_3d(data->status);
+    N_free_array_3d(data->diff_x);
+    N_free_array_3d(data->diff_y);
+    N_free_array_3d(data->diff_z);
+    N_free_array_3d(data->q);
+    N_free_array_3d(data->cs);
+    N_free_array_3d(data->R);
+    N_free_array_3d(data->nf);
+    N_free_array_3d(data->cin);
+
+    N_free_array_3d(data->disp_xx);
+    N_free_array_3d(data->disp_yy);
+    N_free_array_3d(data->disp_zz);
+    N_free_array_3d(data->disp_xy);
+    N_free_array_3d(data->disp_xz);
+    N_free_array_3d(data->disp_yz);
+
+    G_free(data);
+
+    data = NULL;
+
+    return;
+}
+
+/* ************************************************************************* *
+ * ************************************************************************* *
+ * ************************************************************************* */
+/*!
+ * \brief Release the memory of the solute transport data structure in two dimensions
+ *
+ * \param data N_solute_transport_data2d *
+ * \return void *
+ * */
+void N_free_solute_transport_data2d(N_solute_transport_data2d * data)
+{
+    N_free_array_2d(data->c);
+    N_free_array_2d(data->c_start);
+    N_free_array_2d(data->status);
+    N_free_array_2d(data->diff_x);
+    N_free_array_2d(data->diff_y);
+    N_free_array_2d(data->q);
+    N_free_array_2d(data->cs);
+    N_free_array_2d(data->R);
+    N_free_array_2d(data->nf);
+    N_free_array_2d(data->cin);
+    N_free_array_2d(data->top);
+    N_free_array_2d(data->bottom);
+
+    N_free_array_2d(data->disp_xx);
+    N_free_array_2d(data->disp_yy);
+    N_free_array_2d(data->disp_xy);
+
+    G_free(data);
+
+    data = NULL;
+
+    return;
+}
+
+/*!
+ * \brief Compute the transmission boundary condition in 2d
+ *
+ * This function calculates the transmission boundary condition
+ * for each cell with status N_CELL_TRANSMISSION. The surrounding
+ * gradient field is used to verfiy the flow direction. If a flow
+ * goes into a cell, the concentration (data->c) from the neighbour cell is
+ * added to the transmission cell. If the flow from several neighbour 
+ * cells goes into the cell, the concentration mean is calculated.
+ * 
+ * The new concentrations are written into the data->c_start array,
+ * so they can be handled by the matrix assembling function.
+ *
+ * \param data N_solute_transport_data2d *
+ * \return void *
+ * */
+void N_calc_solute_transport_transmission_2d(N_solute_transport_data2d * data)
+{
+    int i, j, count = 1;
+    int cols, rows;
+    double c;
+    N_gradient_2d grad;
+
+    cols = data->grad->cols;
+    rows = data->grad->rows;
+
+    G_debug(2,
+	    "N_calc_solute_transport_transmission_2d: calculating transmission boundary");
+
+    for (j = 0; j < rows; j++) {
+	for (i = 0; i < cols; i++) {
+	    if (N_get_array_2d_d_value(data->status, i, j) ==
+		N_CELL_TRANSMISSION) {
+		count = 0;
+		/*get the gradient neighbours */
+		N_get_gradient_2d(data->grad, &grad, i, j);
+		c = 0;
+		/*
+		   c = N_get_array_2d_d_value(data->c_start, i, j);
+		   if(c > 0)
+		   count++;
+		 */
+
+		if (grad.WC > 0 &&
+		    !N_is_array_2d_value_null(data->c, i - 1, j)) {
+		    c += N_get_array_2d_d_value(data->c, i - 1, j);
+		    count++;
+		}
+		if (grad.EC < 0 &&
+		    !N_is_array_2d_value_null(data->c, i + 1, j)) {
+		    c += N_get_array_2d_d_value(data->c, i + 1, j);
+		    count++;
+		}
+		if (grad.NC < 0 &&
+		    !N_is_array_2d_value_null(data->c, i, j - 1)) {
+		    c += N_get_array_2d_d_value(data->c, i, j - 1);
+		    count++;
+		}
+		if (grad.SC > 0 &&
+		    !N_is_array_2d_value_null(data->c, i, j + 1)) {
+		    c += N_get_array_2d_d_value(data->c, i, j + 1);
+		    count++;
+		}
+		if (count != 0)
+		    c = c / (double)count;
+		/*make sure it is not NAN */
+		if (c > 0 || c == 0 || c < 0)
+		    N_put_array_2d_d_value(data->c_start, i, j, c);
+	    }
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Compute the dispersivity tensor based on the solute transport data in 2d
+ *
+ * The dispersivity tensor is stored in the data structure.
+ * To compute the dispersivity tensor, the dispersivity lentghs and the gradient field
+ * must be present.
+ *
+ * This is just a simple tensor computation which should be extended.
+ *
+ * \todo Change the tensor calculation to a mor realistic algorithm 
+ *
+ * \param data N_solute_transport_data2d *
+ * \return void *
+ * */
+void N_calc_solute_transport_disptensor_2d(N_solute_transport_data2d * data)
+{
+    int i, j;
+    int cols, rows;
+    double vx, vy, vv;
+    double disp_xx, disp_yy, disp_xy;
+    N_gradient_2d grad;
+
+    cols = data->grad->cols;
+    rows = data->grad->rows;
+
+    G_debug(2,
+	    "N_calc_solute_transport_disptensor_2d: calculating the dispersivity tensor");
+
+    for (j = 0; j < rows; j++) {
+	for (i = 0; i < cols; i++) {
+
+	    disp_xx = 0;
+	    disp_yy = 0;
+	    disp_xy = 0;
+
+	    /*get the gradient neighbours */
+	    N_get_gradient_2d(data->grad, &grad, i, j);
+	    vx = (grad.WC + grad.EC) / 2;
+	    vy = (grad.NC + grad.SC) / 2;
+	    vv = sqrt(vx * vx + vy * vy);
+
+	    if (vv != 0) {
+		disp_xx = data->al * vx * vx / vv + data->at * vy * vy / vv;
+		disp_yy = data->at * vx * vx / vv + data->al * vy * vy / vv;
+		disp_xy = (data->al - data->at) * vx * vy / vv;
+	    }
+
+	    G_debug(5,
+		    "N_calc_solute_transport_disptensor_2d: [%i][%i] disp_xx %g disp_yy %g disp_xy %g",
+		    i, j, disp_xx, disp_yy, disp_xy);
+	    N_put_array_2d_d_value(data->disp_xx, i, j, disp_xx);
+	    N_put_array_2d_d_value(data->disp_yy, i, j, disp_yy);
+	    N_put_array_2d_d_value(data->disp_xy, i, j, disp_xy);
+	}
+    }
+
+    return;
+}
+
+/*!
+ * \brief Compute the dispersivity tensor based on the solute transport data in 3d
+ *
+ * The dispersivity tensor is stored in the data structure.
+ * To compute the dispersivity tensor, the dispersivity lentghs and the gradient field
+ * must be present.
+ * 
+ * This is just a simple tensor computation which should be extended.
+ *
+ * \todo Change the tensor calculation to a mor realistic algorithm 
+ *
+ * \param data N_solute_transport_data3d *
+ * \return void *
+ * */
+void N_calc_solute_transport_disptensor_3d(N_solute_transport_data3d * data)
+{
+    int i, j, k;
+    int cols, rows, depths;
+    double vx, vy, vz, vv;
+    double disp_xx, disp_yy, disp_zz, disp_xy, disp_xz, disp_yz;
+    N_gradient_3d grad;
+
+    cols = data->grad->cols;
+    rows = data->grad->rows;
+    depths = data->grad->depths;
+
+    G_debug(2,
+	    "N_calc_solute_transport_disptensor_3d: calculating the dispersivity tensor");
+
+    for (k = 0; k < depths; k++) {
+	for (j = 0; j < rows; j++) {
+	    for (i = 0; i < cols; i++) {
+		disp_xx = 0;
+		disp_yy = 0;
+		disp_zz = 0;
+		disp_xy = 0;
+		disp_xz = 0;
+		disp_yz = 0;
+
+		/*get the gradient neighbours */
+		N_get_gradient_3d(data->grad, &grad, i, j, k);
+		vx = (grad.WC + grad.EC) / 2;
+		vy = (grad.NC + grad.SC) / 2;
+		vz = (grad.BC + grad.TC) / 2;
+		vv = sqrt(vx * vx + vy * vy + vz * vz);
+
+		if (vv != 0) {
+		    disp_xx =
+			data->al * vx * vx / vv + data->at * vy * vy / vv +
+			data->at * vz * vz / vv;
+		    disp_yy =
+			data->at * vx * vx / vv + data->al * vy * vy / vv +
+			data->at * vz * vz / vv;
+		    disp_zz =
+			data->at * vx * vx / vv + data->at * vy * vy / vv +
+			data->al * vz * vz / vv;
+		    disp_xy = (data->al - data->at) * vx * vy / vv;
+		    disp_xz = (data->al - data->at) * vx * vz / vv;
+		    disp_yz = (data->al - data->at) * vy * vz / vv;
+		}
+
+		G_debug(5,
+			"N_calc_solute_transport_disptensor_3d: [%i][%i][%i] disp_xx %g disp_yy %g disp_zz %g  disp_xy %g disp_xz %g disp_yz %g ",
+			i, j, k, disp_xx, disp_yy, disp_zz, disp_xy, disp_xz,
+			disp_yz);
+		N_put_array_3d_d_value(data->disp_xx, i, j, k, disp_xx);
+		N_put_array_3d_d_value(data->disp_yy, i, j, k, disp_yy);
+		N_put_array_3d_d_value(data->disp_zz, i, j, k, disp_zz);
+		N_put_array_3d_d_value(data->disp_xy, i, j, k, disp_xy);
+		N_put_array_3d_d_value(data->disp_xz, i, j, k, disp_xz);
+		N_put_array_3d_d_value(data->disp_yz, i, j, k, disp_yz);
+	    }
+	}
+    }
+
+    return;
+}

Copied: grass/trunk/lib/gpde/n_tools.c (from rev 62429, grass/trunk/lib/gpde/N_tools.c)
===================================================================
--- grass/trunk/lib/gpde/n_tools.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_tools.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,200 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:     	Array managment functions 
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <math.h>
+#include <grass/N_pde.h>
+#include <grass/glocale.h>
+
+
+/*!
+ * \brief Calculate the arithmetic mean of values a and b
+ *
+ * mean = (a+b)/2
+ *
+ * \param a double
+ * \param b double
+ * \return val double
+ * */
+double N_calc_arith_mean(double a, double b)
+{
+    double val = 0;
+
+    val = (a + b) / 2.0;
+
+    return val;
+}
+
+/*!
+ * \brief Calculate the arithmetic mean of the values in vector a
+ * of size n
+ *
+ * n = [0 ... size[
+ * mean =  (a[0] + a[1] + ... + a[n])/size
+ *
+ * \param a double * -- the value vector 
+ * \param size int -- the size of the vector a
+ * \return val double
+ * */
+double N_calc_arith_mean_n(double *a, int size)
+{
+    double val = 0.0;
+    int i;
+
+    for (i = 0; i < size; i++)
+	val += a[i];
+
+    val = (val / (double)size);
+
+    return val;
+}
+
+
+/*!
+ * \brief Calculate the geometrical mean of values a and b
+ *
+ * mean = sqrt(a*b)
+ *
+ * \param a double
+ * \param b double
+ * \return val double
+ * */
+double N_calc_geom_mean(double a, double b)
+{
+    double val = 0;
+
+    val = sqrt(a * b);
+
+    return val;
+}
+
+/*!
+ * \brief Calculate the geometrical mean of the values in vector a
+ * of size n
+ *
+ * n = [0 ... size[
+ * mean =  pow((a[0] * a[1] * ... * a[n]), 1.0/size)
+ *
+ * \param a double * -- the value vector 
+ * \param size int -- the size of the vector a
+ * \return val double
+ * */
+double N_calc_geom_mean_n(double *a, int size)
+{
+    double val = 1;
+    int i;
+
+    for (i = 0; i < size; i++)
+	val *= a[i];
+
+    val = (double)pow((long double)val, (long double)1.0 / (long double)size);
+
+    return val;
+}
+
+
+/*!
+ * \brief Calculate the harmonical mean of values a and b
+ *
+ * mean = 2*(a*b)/(a + b)
+ *
+ * \param a double
+ * \param b double
+ * \return val double -- if (a + b) == 0, a 0 is returned
+ * */
+double N_calc_harmonic_mean(double a, double b)
+{
+    double val = 0.0;
+
+    if ((a + b) != 0)
+	val = 2.0 * (a * b) / (a + b);
+
+    return val;
+}
+
+/*!
+ * \brief Calculate the harmonical mean of the values in vector a
+ * of size n
+ *
+ * n = [0 ... size[
+ * mean = 1/(1/size *(1/a[0] + 1/a[1] + ... + 1/a[n]))
+ * 
+ * \param a double * -- the value vector 
+ * \param size int -- the size of the vector a
+ * \return val double -- if one division with 0 is detected, 0 will be returned
+ * */
+double N_calc_harmonic_mean_n(double *a, int size)
+{
+    double val = 0;
+    int i;
+
+    for (i = 0; i < size; i++)
+	if (a[i] != 0.0)
+	    val += 1.0 / a[i];
+	else
+	    return 0.0;
+
+    if (val == 0.0)
+	return 0.0;
+    else
+	val = 1.0 / (1.0 / (double)size * val);
+
+    return val;
+}
+
+
+/*!
+ * \brief Calculate the quadratic mean of values a and b
+ *
+ * mean = sqrt((a*a + b*b)/2)
+ *
+ * \param a double
+ * \param b double
+ * \return val double 
+ * */
+double N_calc_quad_mean(double a, double b)
+{
+    double val = 0.0;
+
+    val = sqrt((a * a + b * b) / 2.0);
+
+    return val;
+}
+
+/*!
+ * \brief Calculate the quadratic mean of the values in vector a
+ * of size n
+ *
+ * n = [0 ... size[
+ * mean = sqrt((a[0]*a[0] + a[1]*a[1] + ... + a[n]*a[n])/size)
+ * 
+ * \param a double * -- the value vector 
+ * \param size int -- the size of the vector a
+ * \return val double 
+ * */
+double N_calc_quad_mean_n(double *a, int size)
+{
+    double val = 0;
+    int i;
+
+    for (i = 0; i < size; i++)
+	val += a[i] * a[i];
+
+    val = sqrt(val / (double)size);
+
+    return val;
+}

Copied: grass/trunk/lib/gpde/n_upwind.c (from rev 62429, grass/trunk/lib/gpde/N_upwind.c)
===================================================================
--- grass/trunk/lib/gpde/n_upwind.c	                        (rev 0)
+++ grass/trunk/lib/gpde/n_upwind.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,77 @@
+
+/*****************************************************************************
+*
+* MODULE:       Grass PDE Numerical Library
+* AUTHOR(S):    Soeren Gebbert, Berlin (GER) Dec 2006
+* 		soerengebbert <at> gmx <dot> de
+*               
+* PURPOSE:      upwinding stabilization algorithms
+* 		part of the gpde library
+*
+* COPYRIGHT:    (C) 2000 by the GRASS Development Team
+*
+*               This program is free software under the GNU General Public
+*               License (>=v2). Read the file COPYING that comes with GRASS
+*               for details.
+*
+*****************************************************************************/
+
+#include <math.h>
+#include <grass/N_pde.h>
+
+
+/*! \brief full upwinding stabilization algorithm
+ *
+ * The arguments are values to compute the local peclet number
+ *
+ * \param sprod double -- the scalar produkt between the velocity vector and the normal vector between two points
+ * \param distance double -- distance between two points
+ * \param D double -- diffusion/dispersion tensor part between two points
+ *
+ * \return the weighting factor
+ * */
+double N_full_upwinding(double sprod, double distance, double D)
+{
+    double z;
+
+    if (D == 0)
+	return 0.5;
+
+    /*compute the local peclet number */
+    z = sprod * distance / D;
+
+    if (z > 0)
+	return 1;
+    if (z == 0)
+	return 0.5;
+    if (z < 0)
+	return 0;
+
+    return 0;
+}
+
+/*! \brief exponential upwinding stabilization algorithm
+ *
+ * The arguments are values to compute the local peclet number
+ *
+ * \param sprod double -- the scalar produkt between the velocity vector and the normal vector between two points
+ * \param distance double -- distance between two points
+ * \param D double -- diffusion/dispersion tensor part between two points
+ *
+ * \return the weighting factor
+ * */
+double N_exp_upwinding(double sprod, double distance, double D)
+{
+    double z;
+
+    if (D == 0)
+	return 0.5;
+
+    /*compute the local peclet number */
+    z = sprod * distance / D;
+
+    if (z != 0)
+	return (1 - (1 / z) * (1 - (z / (exp(z) - 1))));
+
+    return 0.5;
+}

Deleted: grass/trunk/lib/ogsf/GK2.c
===================================================================
--- grass/trunk/lib/ogsf/GK2.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GK2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,625 +0,0 @@
-/*!
-   \file GK2.c
-
-   \brief OGSF library - setting and manipulating keyframes animation
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown USACERL, GMSL/University of Illinois
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <stdlib.h>
-
-#include <grass/gis.h>
-#include <grass/glocale.h>
-#include <grass/ogsf.h>
-
-static int _add_key(Keylist *, int, float);
-static void _remove_key(Keylist *);
-
-static Keylist *Keys = NULL;
-static Keylist *Keytail = NULL;
-static Viewnode *Views = NULL;
-static float Keystartpos = 0.0;
-static float Keyendpos = 1.0;
-static float Tension = 0.8;
-static int Viewsteps = 0;
-static int Numkeys = 0;
-static int Interpmode = KF_SPLINE;
-static int Fmode = 0;
-
-/* next & prior already initialized to NULL */
-static int _add_key(Keylist * newk, int force_replace, float precis)
-{
-    Keylist *k, *tempk, *prev;
-    int found;
-
-    found = 0;
-    prev = NULL;
-
-    /* if(Viewsteps) precis = 0.5/Viewsteps; */
-    for (k = Keys; k; k = k->next) {
-	if (k->pos >= newk->pos - precis && k->pos <= newk->pos + precis) {
-	    if (force_replace) {
-
-		if (k->prior) {
-		    k->prior->next = newk;
-		    newk->prior = prev;
-		}
-		else {
-		    Keys = newk;
-		}
-
-		newk->next = k->next;
-		newk->prior = k->prior;
-		tempk = k;
-		k = newk;
-		free(tempk);
-	    }
-	    else {
-		free(newk);
-	    }
-
-	    return (-1);
-	}
-    }
-
-    if (Keys) {
-	if (newk->pos < Keys->pos) {
-	    /* new will be first */
-	    newk->next = Keys;
-	    Keys->prior = newk;
-	    Keys = newk;
-	}
-	else {
-	    prev = k = Keys;
-	    while (k && !found) {
-		if (k->pos > newk->pos) {
-		    prev->next = newk;
-		    newk->next = k;
-		    newk->prior = prev;
-		    k->prior = newk;
-		    found = 1;
-		}
-
-		prev = k;
-		k = k->next;
-	    }
-	    if (!found) {
-		Keytail = prev->next = newk;
-		newk->prior = prev;
-	    }
-	}
-    }
-    else {
-	Keys = Keytail = newk;
-    }
-
-    ++Numkeys;
-    return (1);
-}
-
-static void _remove_key(Keylist * k)
-{
-    if (k->prior) {
-	k->prior->next = k->next;
-	if (k->next) {
-	    k->next->prior = k->prior;
-	}
-	else {
-	    Keytail = k->prior;
-	}
-    }
-    else {
-	Keys = k->next;
-	if (k->next) {
-	    k->next->prior = NULL;
-	}
-    }
-    k->next = k->prior = NULL;
-
-    return;
-}
-
-/*!
-   \brief Set interpolation mode 
-
-   \param mode interpolation mode (KF_LINEAR or KF_SPLINE)
-
-   \return 1 on success
-   \return -1 on error (invalid interpolation mode)
- */
-int GK_set_interpmode(int mode)
-{
-    if (KF_LEGAL_MODE(mode)) {
-	Interpmode = mode;
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set value for tension when interpmode is KF_SPLINE. 
-
-   \param tens value tens should be between 0.0; 1.0.
- */
-void GK_set_tension(float tens)
-{
-    Tension = tens > 1.0 ? 1.0 : (tens < 0.0 ? 0.0 : tens);
-
-    /* for now */
-    if (Views) {
-	GK_update_frames();
-	GS_set_draw(GSD_BACK);
-	GS_ready_draw();
-	GS_clear(GS_background_color());
-	GS_alldraw_wire();
-
-	gk_draw_path(Views, Viewsteps, Keys);
-
-	GS_done_draw();
-    }
-
-    return;
-}
-
-void GK_showtension_start(void)
-{
-    return;
-}
-
-/*!
-   \brief Show tension stop ?
-
-   Use GK_showtension_start/GK_update_tension/GK_showtension_stop to
-   initialize and stop multi-view display of path when changing
-   tension.
- */
-void GK_showtension_stop(void)
-{
-    return;
-}
-
-/*!
-   \brief Update tension
- */
-void GK_update_tension(void)
-{
-    if (Views) {
-	GK_update_frames();
-    }
-
-    return;
-}
-
-/*!
-   \brief Print keyframe info
-
-   \param name filename
- */
-void GK_print_keys(const char *name)
-{
-    Keylist *k;
-    FILE *fp;
-    int cnt = 1;
-
-    if (NULL == (fp = fopen(name, "w"))) {
-	G_fatal_error(_("Unable to open file <%s> for writing"), name);
-    }
-    /* write a default frame rate of 30 at top of file */
-    fprintf(fp, "30 \n");
-
-    for (k = Keys; k; k = k->next) {
-
-	fprintf(fp,
-		"{%f {{FromX %f} {FromY %f} {FromZ %f} {DirX %f} {DirY %f} {DirZ %f} {FOV %f} {TWIST %f} {cplane-0 {{pos_x 0.000000} {pos_y 0.000000} {pos_z 0.000000} {blend_type OFF} {rot 0.000000} {tilt 0.000000}}}} keyanimtag%d 0} ",
-		k->pos, k->fields[KF_FROMX], k->fields[KF_FROMY],
-		k->fields[KF_FROMZ], k->fields[KF_DIRX], k->fields[KF_DIRY],
-		k->fields[KF_DIRZ], k->fields[KF_FOV] / 10.,
-		k->fields[KF_TWIST], cnt);
-	cnt++;
-    }
-
-    fclose(fp);
-    return;
-
-}
-
-/*!
-   \brief Recalculate path using the current number of frames requested.
-
-   Call after changing number of frames or when
-   Keyframes change.
- */
-void GK_update_frames(void)
-{
-    Keylist *k;
-    int loop = 0;
-
-    if (Keys) {
-	if (Numkeys > 1) {
-	    k = Keytail;
-	    Keyendpos = k->pos;
-
-	    if (k->fields[KF_FROMX] == Keys->fields[KF_FROMX] &&
-		k->fields[KF_FROMY] == Keys->fields[KF_FROMY] &&
-		k->fields[KF_FROMZ] == Keys->fields[KF_FROMZ]) {
-		loop = 1;
-	    }
-	}
-
-	Keystartpos = Keys->pos;
-    }
-
-    if (Interpmode == KF_LINEAR && Numkeys > 1) {
-	if (Views) {
-	    free(Views);
-	    Views = NULL;
-	}
-
-	Views = gk_make_linear_framesfromkeys(Keys, Numkeys, Viewsteps, loop);
-
-	if (!Views) {
-	    G_warning(_("Check no. of frames requested and keyframes marked"));
-	}
-    }
-    else if (Numkeys > 2) {
-	if (Views) {
-	    free(Views);
-	    Views = NULL;
-	}
-
-	Views = gk_make_framesfromkeys
-	    (Keys, Numkeys, Viewsteps, loop, 1.0 - Tension);
-
-	if (!Views) {
-	    G_warning(_("Check no. of frames requested and keyframes marked"));
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Set the number of frames to be interpolated from keyframes
-
-   \param newsteps number of frames
- */
-void GK_set_numsteps(int newsteps)
-{
-    Viewsteps = newsteps;
-    GK_update_frames();
-
-    return;
-}
-
-/*!
-   \brief Deletes all keyframes, resets field masks.
-
-   Doesn't change number of frames requested.
- */
-void GK_clear_keys(void)
-{
-    gk_free_key(Keys);
-    Keys = NULL;
-    Numkeys = 0;
-    free(Views);
-    Views = NULL;
-
-    Keystartpos = 0.0;
-    Keyendpos = 1.0;
-
-    return;
-}
-
-/*!
-   \brief Move keyframe
-
-   Precis works as in other functions - to identify keyframe to move.
-   Only the first keyframe in the precis range will be moved.
-
-   \param oldpos old position
-   \param precis precision value
-   \param newpos new position
-
-   \return number of keys moved (1 or 0)
- */
-int GK_move_key(float oldpos, float precis, float newpos)
-{
-    Keylist *k;
-
-    for (k = Keys; k; k = k->next) {
-	if (k->pos >= oldpos - precis && k->pos <= oldpos + precis) {
-	    _remove_key(k);
-	    k->pos = newpos;
-	    _add_key(k, 1, precis);
-	    GK_update_frames();
-	    return (1);
-	}
-    }
-
-    return (0);
-}
-
-/*!
-   Delete keyframe
-
-   The values pos and precis are used to determine which keyframes to
-   delete.  Any keyframes with their position within precis of pos will
-   be deleted if justone is zero.  If justone is non-zero, only the first
-   (lowest pos) keyframe in the range will be deleted.
-
-   \param pos position
-   \param precis precision
-   \param justone delete only one keyframe
-
-   \return number of keys deleted.
- */
-int GK_delete_key(float pos, float precis, int justone)
-{
-    Keylist *k, *next;
-    int cnt;
-
-    for (cnt = 0, k = Keys; k;) {
-	next = k->next;
-
-	if (k->pos >= pos - precis && k->pos <= pos + precis) {
-	    cnt++;
-	    _remove_key(k);
-	    free(k);
-	    if (justone) {
-		break;
-	    }
-	}
-
-	k = next;
-    }
-
-    GK_update_frames();
-    return (cnt);
-}
-
-/*!
-   \brief Add keyframe
-
-   The pos value is the relative position in the animation for this
-   particular keyframe - used to compare relative distance to neighboring
-   keyframes, it can be any floating point value.
-
-   The fmask value can be any of the following or'd together:    
-   - KF_FROMX_MASK    
-   - KF_FROMY_MASK    
-   - KF_FROMZ_MASK    
-   - KF_FROM_MASK (KF_FROMX_MASK | KF_FROMY_MASK | KF_FROMZ_MASK) 
-
-   - KF_DIRX_MASK    
-   - KF_DIRY_MASK    
-   - KF_DIRZ_MASK    
-   - KF_DIR_MASK (KF_DIRX_MASK | KF_DIRY_MASK | KF_DIRZ_MASK) 
-
-   - KF_FOV_MASK    
-   - KF_TWIST_MASK    
-
-   - KF_ALL_MASK (KF_FROM_MASK | KF_DIR_MASK | KF_FOV_MASK | KF_TWIST_MASK) 
-
-   Other fields will be added later.
-
-   The value precis and the boolean force_replace are used to determine
-   if a keyframe should be considered to be at the same position as a
-   pre-existing keyframe. e.g., if anykey.pos - newkey.pos <= precis,
-   GK_add_key() will fail unless force_replace is TRUE.
-
-   \param pos postion
-   \param fmaks
-   \param force_replace
-   \param precis precision value
-
-   \return 1 if key is added
-   \return -1 key not added
- */
-int GK_add_key(float pos, unsigned long fmask, int force_replace,
-	       float precis)
-{
-    Keylist *newk;
-    float tmp[3];
-
-    if (NULL == (newk = (Keylist *) malloc(sizeof(Keylist)))) {
-	fprintf(stderr, "Out of memory\n");
-	return (-1);
-    }
-
-    /* All fields set, don't use mask until making Views */
-
-    GS_get_from(tmp);
-    newk->fields[KF_FROMX] = tmp[X];
-    newk->fields[KF_FROMY] = tmp[Y];
-    newk->fields[KF_FROMZ] = tmp[Z];
-
-    G_debug(3, "KEY FROM: %f %f %f", tmp[X], tmp[Y], tmp[Z]);
-
-    /* Instead of View Dir try get_focus (view center) */
-    /* View Dir is implied from eye and center position */
-    /*    GS_get_viewdir(tmp); */
-
-    /* ACS 1 line: was      GS_get_focus(tmp);
-       with this kanimator works also for flythrough navigation
-       also changed in gk.c
-     */
-    GS_get_viewdir(tmp);
-    newk->fields[KF_DIRX] = tmp[X];
-    newk->fields[KF_DIRY] = tmp[Y];
-    newk->fields[KF_DIRZ] = tmp[Z];
-
-    newk->fields[KF_FOV] = GS_get_fov();
-    newk->fields[KF_TWIST] = GS_get_twist();
-    newk->pos = pos;
-    newk->fieldmask = fmask;
-    newk->next = NULL;
-    newk->prior = NULL;
-
-    if (0 < _add_key(newk, force_replace, precis)) {
-	GK_update_frames();
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Moves the animation to frame number "step".
-
-   Step should be a value between 1 and the number of frames.  If
-   render is non-zero, calls draw_all.
-
-   \param step step value
-   \param render
- */
-void GK_do_framestep(int step, int render)
-{
-    if (Views) {
-	if (step > 0 && step <= Viewsteps) {
-	    gk_follow_frames(Views, Viewsteps, Keys, step, 1, render, Fmode);
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw the current path
-
-   \param flag
- */
-void GK_show_path(int flag)
-{
-    if (flag) {
-	Fmode |= FM_PATH;
-
-	if (Views) {
-	    GS_set_draw(GSD_FRONT);
-	    GS_ready_draw();
-
-	    gk_draw_path(Views, Viewsteps, Keys);
-
-	    GS_done_draw();
-
-	}
-    }
-    else {
-	Fmode &= ~FM_PATH;
-    }
-
-    return;
-}
-
-/*!
-   \brief Show vector sets
-
-   \param flag
- */
-void GK_show_vect(int flag)
-{
-    if (flag) {
-	Fmode |= FM_VECT;
-	if (Views) {
-
-	    GS_set_draw(GSD_FRONT);
-	    GS_ready_draw();
-
-	    GV_alldraw_vect();
-
-	    GS_done_draw();
-	}
-    }
-    else {
-	Fmode &= ~FM_VECT;
-    }
-
-    return;
-}
-
-/*!
-   \brief Show point sets
-
-   \param flag
- */
-void GK_show_site(int flag)
-{
-    if (flag) {
-	Fmode |= FM_SITE;
-
-	if (Views) {
-
-	    GS_set_draw(GSD_FRONT);
-	    GS_ready_draw();
-
-	    GP_alldraw_site();
-
-	    GS_done_draw();
-
-	}
-    }
-    else {
-	Fmode &= ~FM_SITE;
-    }
-
-    return;
-}
-
-/*!
-   \brief Show volumes
-
-   \param flag
- */
-void GK_show_vol(int flag)
-{
-    if (flag) {
-	Fmode |= FM_VOL;
-
-	if (Views) {
-
-	    GS_set_draw(GSD_FRONT);
-	    GS_ready_draw();
-
-	    GVL_alldraw_vol();
-
-	    GS_done_draw();
-
-	}
-    }
-    else {
-	Fmode &= ~FM_VOL;
-    }
-
-    return;
-}
-
-/*!
-   \brief Show list
-
-   \param flag
- */
-void GK_show_list(int flag)
-{
-    if (flag) {
-	Fmode |= FM_LABEL;
-
-	if (Views) {
-	    GS_draw_all_list();
-	}
-    }
-    else {
-	Fmode &= ~FM_LABEL;
-    }
-
-    return;
-}

Deleted: grass/trunk/lib/ogsf/GP2.c
===================================================================
--- grass/trunk/lib/ogsf/GP2.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GP2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,710 +0,0 @@
-/*!
-   \file lib/ogsf/GP2.c
-
-   \brief OGSF library - loading and manipulating point sets (higher level functions)
-
-   (C) 1999-2008, 2011 by the GRASS Development Team
-
-   This program is free software under the GNU General Public License
-   (>=v2). Read the file COPYING that comes with GRASS for details.
-
-   \author Bill Brown USACERL (January 1994)
-   \author Updated by Martin landa <landa.martin gmail.com>
-   (doxygenized in May 2008, thematic mapping in June 2011)
- */
-
-#include <stdlib.h>
-#include <string.h>
-
-#include <grass/gis.h>
-#include <grass/ogsf.h>
-#include <grass/glocale.h>
-
-#include "gsget.h"
-
-static int Site_ID[MAX_SITES];
-static int Next_site = 0;
-
-/*!
-   \brief Check if point set exists
-
-   \param id point set id
-
-   \return 1 found
-   \return 0 not found
- */
-int GP_site_exists(int id)
-{
-    int i, found = 0;
-
-    G_debug(4, "GP_site_exists(%d)", id);
-
-    if (NULL == gp_get_site(id)) {
-	return 0;
-    }
-
-    for (i = 0; i < Next_site && !found; i++) {
-	if (Site_ID[i] == id) {
-	    found = 1;
-	}
-    }
-
-    G_debug(3, "GP_site_exists(): found=%d", found);
-
-    return found;
-}
-
-/*!
-   \brief Create new point set
-
-   \return point set id
-   \return -1 on error (number of point sets exceeded)
- */
-int GP_new_site(void)
-{
-    geosite *np;
-
-    if (Next_site < MAX_SITES) {
-	np = gp_get_new_site();
-	gp_set_defaults(np);
-	Site_ID[Next_site] = np->gsite_id;
-	++Next_site;
-
-	G_debug(3, "GP_new_site() id=%d", np->gsite_id);
-
-	return np->gsite_id;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Get number of loaded point sets
-
-   \return number of point sets
- */
-int GP_num_sites(void)
-{
-    return gp_num_sites();
-}
-
-/*!
-   \brief Get list of point sets
-
-   Must freed when no longer needed!
-
-   \param numsites number of point sets
-
-   \return pointer to list of points sets
-   \return NULL on error
- */
-int *GP_get_site_list(int *numsites)
-{
-    int i, *ret;
-
-    *numsites = Next_site;
-
-    if (Next_site) {
-	ret = (int *)G_malloc(Next_site * sizeof(int));	/* G_fatal_error */
-	if (!ret) {
-	    return NULL;
-	}
-
-	for (i = 0; i < Next_site; i++) {
-	    ret[i] = Site_ID[i];
-	}
-
-	return ret;
-    }
-
-    return NULL;
-}
-
-/*!
-   \brief Delete registrated point set
-
-   \param id point set id
-
-   \return 1 on success
-   \return -1 on error (point sets not available)
- */
-int GP_delete_site(int id)
-{
-    int i, j, found = 0;
-
-    G_debug(4, "GP_delete_site(%d)", id);
-
-    if (GP_site_exists(id)) {
-	gp_delete_site(id);
-
-	for (i = 0; i < Next_site && !found; i++) {
-	    if (Site_ID[i] == id) {
-		found = 1;
-		for (j = i; j < Next_site; j++) {
-		    Site_ID[j] = Site_ID[j + 1];
-		}
-	    }
-	}
-
-	if (found) {
-	    --Next_site;
-	    return 1;
-	}
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Load point set from file
-
-   Check to see if handle already loaded, if so - free before loading
-   new for now, always load to memory.
-
-   \todo load file handle & ready for reading instead of using memory
-
-   \param id point set id
-   \param filename point set filename
-
-   \return -1 on error
-   \return 1 on success
- */
-int GP_load_site(int id, const char *filename)
-{
-    geosite *gp;
-
-    G_debug(3, "GP_load_site(id=%d, name=%s)", id, filename);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    if (gp->points) {
-	gp_free_sitemem(gp);
-    }
-
-    gp->filename = G_store(filename);
-
-    gp->points = Gp_load_sites(filename, &(gp->n_sites), &(gp->has_z));
-    
-    if (gp->points) {
-	return 1;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Get point set filename
-
-   Note: char array is allocated by G_store()
-
-   \param id point set id
-   \param[out] filename point set filename
-
-   \return -1 on error (point set not found)
-   \return 1 on success
- */
-int GP_get_sitename(int id, char **filename)
-{
-    geosite *gp;
-
-    G_debug(4, "GP_get_sitename(%d)", id);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    *filename = G_store(gp->filename);
-
-    return 1;
-}
-
-/*!
-   \brief Get point set style
-
-   \param id point set id
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GP_get_style(int id, int *color, int *width, float *size, int *symbol)
-{
-    geosite *gp;
-
-    G_debug(4, "GP_get_style(%d)", id);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    *color = gp->style->color;
-    *width = gp->style->width;
-    *symbol = gp->style->symbol;
-    *size = gp->style->size;
-
-    return 1;
-}
-
-/*!
-   \brief Set point style
-
-   Supported icon symbols (markers):
-    - ST_X
-    - ST_BOX
-    - ST_SPHERE
-    - ST_CUBE
-    - ST_DIAMOND
-    - ST_DEC_TREE
-    - ST_CON_TREE
-    - ST_ASTER
-    - ST_GYRO
-    - ST_HISTOGRAM
-
-   \param id point set id
-   \param color icon color
-   \param width icon line width
-   \param size icon size
-   \param symbol icon symbol
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GP_set_style(int id, int color, int width, float size, int symbol)
-{
-    geosite *gp;
-
-    G_debug(4, "GP_set_style(id=%d, color=%d, width=%d, size=%f, symbol=%d)", id, color, width, size,
-	    symbol);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    gp->style->color = color;
-    gp->style->symbol = symbol;
-    gp->style->size = size;
-    gp->style->width = width;
-
-    return 1;
-}
-
-/*!
-   \brief Set point set style for thematic mapping
-
-   Updates also style for each geopoint.
-   
-   \param id point set id
-   \param layer layer number for thematic mapping (-1 for undefined)
-   \param color icon color column name
-   \param width icon line width column name
-   \param size icon size column name
-   \param symbol icon symbol column name
-   \param colors pointer to Colors structure or NULL
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GP_set_style_thematic(int id, int layer, const char* color, const char* width,
-			  const char* size, const char* symbol, struct Colors *color_rules)
-{
-    geosite *gp;
-    
-    G_debug(4, "GP_set_style_thematic(id=%d, layer=%d, color=%s, width=%s, size=%s, symbol=%s)", id, layer,
-	    color, width, size, symbol);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    if(!gp->tstyle)
-	gp->tstyle = (gvstyle_thematic *)G_malloc(sizeof(gvstyle_thematic));
-    G_zero(gp->tstyle, sizeof(gvstyle_thematic));
-    
-    gp->tstyle->active = 1;
-    gp->tstyle->layer = layer;
-    if (color)
-	gp->tstyle->color_column = G_store(color);
-    if (symbol)
-	gp->tstyle->symbol_column = G_store(symbol);
-    if (size)
-	gp->tstyle->size_column = G_store(size);
-    if (width)
-	gp->tstyle->width_column = G_store(width);
-
-    Gp_load_sites_thematic(gp, color_rules);
-
-    return 1;
-}
-
-/*!
-   \brief Make style for thematic mapping inactive
-   
-   \param id point set id
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GP_unset_style_thematic(int id)
-{
-    geosite *gp;
-
-    G_debug(4, "GP_unset_style_thematic(): id=%d", id);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    if (gp->tstyle) {
-	gp->tstyle->active = 0;
-    }
-
-    return 1;
-}
-
-/*!
-   \brief Set z mode for point set
-
-   \param id point set id
-   \param use_z TRUE to use z-coordinaces when vector map is 3D
-
-   \return 1 on success
-   \return 0 vector map is not 3D
-   \return -1 on error (invalid point set id)
- */
-/* I don't see who is using it? Why it's required? */
-int GP_set_zmode(int id, int use_z)
-{
-    geosite *gp;
-
-    G_debug(3, "GP_set_zmode(%d,%d)", id, use_z);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    if (use_z) {
-	if (gp->has_z) {
-	    gp->use_z = 1;
-	    return 1;
-	}
-
-	return 0;
-    }
-
-    gp->use_z = 0;
-    return 1;
-}
-
-/*!
-   \brief Get z-mode
-
-   \todo Who's using this?
-   
-   \param id point set id
-   \param[out] use_z non-zero code to use z
-
-   \return -1 on error (invalid point set id)
-   \return 1 on success
- */
-int GP_get_zmode(int id, int *use_z)
-{
-    geosite *gp;
-
-    G_debug(4, "GP_get_zmode(%d)", id);
-
-    if (NULL == (gp = gp_get_site(id))) {
-	return -1;
-    }
-
-    *use_z = gp->use_z;
-    return 1;
-}
-
-/*!
-   \brief Set transformation params
-
-   \param id point set id
-   \param xtrans,ytrans,ztrans x/y/z values
- */
-void GP_set_trans(int id, float xtrans, float ytrans, float ztrans)
-{
-    geosite *gp;
-
-    G_debug(3, "GP_set_trans(): id=%d trans=%f,%f,%f",
-	    id, xtrans, ytrans, ztrans);
-
-    gp = gp_get_site(id);
-    if (gp) {
-	gp->x_trans = xtrans;
-	gp->y_trans = ytrans;
-	gp->z_trans = ztrans;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get transformation params
-
-   \param id point set id
-   \param[out] xtrans,ytrans,ztrans x/y/z values
- */
-void GP_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
-{
-    geosite *gp;
-
-    gp = gp_get_site(id);
-
-    if (gp) {
-	*xtrans = gp->x_trans;
-	*ytrans = gp->y_trans;
-	*ztrans = gp->z_trans;
-    }
-
-    G_debug(3, "GP_get_trans(): id=%d, trans=%f,%f,%f",
-	    id, *xtrans, *ytrans, *ztrans);
-
-    return;
-}
-
-/*!
-   \brief Select surface for given point set
-
-   \param hp point set id
-   \param hs surface id
-
-   \return 1 surface selected
-   \return -1 on error
- */
-int GP_select_surf(int hp, int hs)
-{
-    geosite *gp;
-
-    G_debug(3, "GP_select_surf(%d,%d)", hp, hs);
-
-    if (GP_surf_is_selected(hp, hs)) {
-	return 1;
-    }
-
-    gp = gp_get_site(hp);
-
-    if (gp && GS_surf_exists(hs)) {
-	gp->drape_surf_id[gp->n_surfs] = hs;
-	gp->n_surfs += 1;
-	return 1;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Unselect surface
-
-   \param hp point set id
-   \param hs surface id
-
-   \return 1 surface unselected
-   \return -1 on error
- */
-int GP_unselect_surf(int hp, int hs)
-{
-    geosite *gp;
-    int i, j;
-
-    G_debug(3, "GP_unselect_surf(%d,%d)", hp, hs);
-
-    if (!GP_surf_is_selected(hp, hs)) {
-	return 1;
-    }
-
-    gp = gp_get_site(hp);
-
-    if (gp) {
-	for (i = 0; i < gp->n_surfs; i++) {
-	    if (gp->drape_surf_id[i] == hs) {
-		for (j = i; j < gp->n_surfs - 1; j++) {
-		    gp->drape_surf_id[j] = gp->drape_surf_id[j + 1];
-		}
-
-		gp->n_surfs -= 1;
-		return 1;
-	    }
-	}
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Check if surface is selected
-
-   \param hp point set id
-   \param hs surface id
-
-   \return 1 selected
-   \return 0 not selected
- */
-int GP_surf_is_selected(int hp, int hs)
-{
-    int i;
-    geosite *gp;
-
-    G_debug(3, "GP_surf_is_selected(%d,%d)", hp, hs);
-
-    gp = gp_get_site(hp);
-
-    if (gp) {
-	for (i = 0; i < gp->n_surfs; i++) {
-	    if (hs == gp->drape_surf_id[i]) {
-		return 1;
-	    }
-	}
-    }
-
-    return 0;
-}
-
-/*!
-   \brief Draw point set
-
-   \param id point set id
- */
-void GP_draw_site(int id)
-{
-    geosurf *gs;
-    geosite *gp;
-    int i;
-    float n, yo, xo, e;
-
-    gp = gp_get_site(id);
-    GS_get_region(&n, &yo, &xo, &e);
-
-    /* kind of sloppy - maybe site files should have an origin, too */
-    if (gp) {
-	if (gp->use_z && gp->has_z) {
-	    gpd_3dsite(gp, xo, yo, 0);
-	}
-	else {
-	    for (i = 0; i < gp->n_surfs; i++) {
-		gs = gs_get_surf(gp->drape_surf_id[i]);
-
-		if (gs) {
-		    gpd_2dsite(gp, gs, 0);
-		    G_debug(5, "Drawing site %d on Surf %d", id,
-			    gp->drape_surf_id[i]);
-		}
-	    }
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all available point sets
- */
-void GP_alldraw_site(void)
-{
-    int id;
-
-    for (id = 0; id < Next_site; id++) {
-	GP_draw_site(Site_ID[id]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Set client data
-
-   \param id point set id
-   \param clientd client data
-
-   \return 1 on success
-   \return -1 on error (invalid point set id)
- */
-int GP_Set_ClientData(int id, void *clientd)
-{
-    geosite *gp;
-
-    gp = gp_get_site(id);
-
-    if (gp) {
-	gp->clientdata = clientd;
-	return 1;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Get client data
-
-   \param id point set id
-
-   \return pointer to client data
-   \return NULL on error
- */
-void *GP_Get_ClientData(int id)
-{
-    geosite *gp;
-
-    gp = gp_get_site(id);
-    if (gp) {
-	return (gp->clientdata);
-    }
-
-    return NULL;
-}
-
-/*!
-  \brief Determine point marker symbol for string
-
-  Supported markers:
-    - ST_X
-    - ST_BOX
-    - ST_SPHERE
-    - ST_CUBE
-    - ST_DIAMOND
-    - ST_DEC_TREE
-    - ST_CON_TREE
-    - ST_ASTER
-    - ST_GYRO
-    - ST_HISTOGRAM
-
-  \param str string buffer
-
-  \return marker code (default: ST_SPHERE)
-*/
-int GP_str_to_marker(const char *str)
-{
-    int marker;
-
-    if (strcmp(str, "x") == 0)
-	marker = ST_X;
-    else if (strcmp(str, "box") == 0)
-	marker = ST_BOX;
-    else if (strcmp(str, "sphere") == 0)
-	marker = ST_SPHERE;
-    else if (strcmp(str, "cube") == 0)
-	marker = ST_CUBE;
-    else if (strcmp(str, "diamond") == 0)
-	marker = ST_DIAMOND;
-    else if (strcmp(str, "dec_tree") == 0)
-	marker = ST_DEC_TREE;
-    else if (strcmp(str, "con_tree") == 0)
-	marker = ST_CON_TREE;
-    else if (strcmp(str, "aster") == 0)
-	marker = ST_ASTER;
-    else if (strcmp(str, "gyro") == 0)
-	marker = ST_GYRO;
-    else if (strcmp(str, "histogram") == 0)
-	marker = ST_HISTOGRAM;
-    else {
-	G_warning(_("Unknown icon marker, using \"sphere\""));
-	marker = ST_SPHERE;
-    }
-
-    return marker;
-}

Deleted: grass/trunk/lib/ogsf/GS2.c
===================================================================
--- grass/trunk/lib/ogsf/GS2.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GS2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,3495 +0,0 @@
-/*!
-   \file GS2.c
-
-   \brief OGSF library - loading and manipulating surfaces (higher level functions)
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   Plans for handling color maps:
-   NOW:
-   if able to load as unsigned char, make lookup table containing palette
-   otherwise, load directly as packed color, set lookup = NULL
-   MAYBE LATER:
-   if able to load as POSITIVE short, make lookup table containing palette
-   - may want to calculate savings first (ie,  numcells > 32768)
-   (not exactly, it's Friday & time to go home - figure it later)
-   otherwise, load directly as packed color, set lookup = NULL
-   MESSY! - need to fix up!
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown USACERL (1993)
-   \author Pierre de Mouveaux <p_de_mouveaux hotmail.com> (updated October 1999)
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <stdlib.h>
-#include <string.h>
-#include <math.h>
-
-#include <grass/config.h>
-
-#if defined(OPENGL_X11) || defined(OPENGL_WINDOWS)
-#include <GL/gl.h>
-#include <GL/glu.h>
-#elif defined(OPENGL_AQUA)
-#include <OpenGL/gl.h>
-#include <OpenGL/glu.h>
-#endif
-
-#include <grass/gis.h>
-#include <grass/raster.h>
-#include <grass/ogsf.h>
-#include <grass/glocale.h>
-
-#include "gsget.h"
-#include "rowcol.h"
-#include "rgbpack.h"
-
-/* Hack to make NVIZ2.2 query functions.("What's Here" and "Look at")
- * to work.
- * Uses gs_los_intersect1() instead of gs_los_intersect().
- * Pierre de Mouveaux - 31 oct. 1999. p_de_mouveaux at hotmail.com.
- */
-#define NVIZ_HACK 1
-
-int gsd_getViewport(GLint *, GLint *);
-
-/* array of surface ids */
-static int Surf_ID[MAX_SURFS];
-static int Next_surf = 0;
-static int SDref_surf = 0;
-
-/* attributes array */
-static float Default_const[MAX_ATTS];
-static float Default_nulls[MAX_ATTS];
-
-/* largest dimension */
-static float Longdim;
-
-/* N, S, W, E */
-static float Region[4];
-static geoview Gv;
-static geodisplay Gd;
-static struct Cell_head wind;
-static int Buffermode;
-static int Numlights = 0;
-static int Resetlight = 1;
-static int Modelshowing = 0;
-
-void void_func(void)
-{
-    return;
-}
-
-/*!
-   \brief Initialize OGSF library
-
-   Get region settings - wind
-
-   Set Region (NSWE array) and compute scale
- */
-void GS_libinit(void)
-{
-    static int first = 1;
-
-    G_get_set_window(&wind);
-
-    Region[0] = wind.north;
-    Region[1] = wind.south;
-    Region[2] = wind.west;
-    Region[3] = wind.east;
-
-    /* scale largest dimension to GS_UNIT_SIZE */
-    if ((wind.east - wind.west) > (wind.north - wind.south)) {
-	Longdim = (wind.east - wind.west);
-    }
-    else {
-	Longdim = (wind.north - wind.south);
-    }
-
-    Gv.scale = GS_UNIT_SIZE / Longdim;
-
-    G_debug(1, "GS_libinit(): n=%f s=%f w=%f e=%f scale=%f first=%d",
-	    Region[0], Region[1], Region[2], Region[3], Gv.scale, first);
-    
-    Cxl_func = void_func;
-    Swap_func = void_func;
-
-    
-    if (first) {
-	gs_init();
-    }
-
-    first = 0;
-
-    return;
-}
-
-/*!
-   \brief Get largest dimension
-
-   \param[out] dim dimension
-
-   \return 1
- */
-int GS_get_longdim(float *dim)
-{
-    *dim = Longdim;
-
-    G_debug(3, "GS_get_longdim(): dim=%g", *dim);
-
-    return (1);
-}
-
-/*!
-   \brief Get 2D region extent
-
-   \param[out] n,s,w,e extent values
-
-   \return 1
- */
-int GS_get_region(float *n, float *s, float *w, float *e)
-{
-    *n = Region[0];
-    *s = Region[1];
-    *w = Region[2];
-    *e = Region[3];
-
-    return (1);
-}
-
-/*!
-   \brief Set default attributes for map objects
-
-   \param defs attributes array (dim MAX_ATTS)
-   \param null_defs null attributes array (dim MAX_ATTS)
- */
-void GS_set_att_defaults(float *defs, float *null_defs)
-{
-    int i;
-
-    G_debug(3, "GS_set_att_defaults");
-
-    for (i = 0; i < MAX_ATTS; i++) {
-	Default_const[i] = defs[i];
-	Default_nulls[i] = null_defs[i];
-    }
-
-    return;
-}
-
-/*!
-   Check if surface exists
-
-   \param id surface id
-
-   \return 0 not found
-   \return 1 found
- */
-int GS_surf_exists(int id)
-{
-    int i, found = 0;
-
-    G_debug(3, "GS_surf_exists(): id=%d", id);
-
-
-    if (NULL == gs_get_surf(id)) {
-	return (0);
-    }
-
-    for (i = 0; i < Next_surf && !found; i++) {
-	if (Surf_ID[i] == id) {
-	    found = 1;
-	}
-    }
-
-    return (found);
-}
-
-/*!
-   \brief Add new surface
-
-   Note that origin has 1/2 cell added to represent center of cells
-   because library assumes that east - west = (cols - 1) * ew_res,
-   since left and right columns are on the edges.
-
-   \return surface id
-   \return -1 on error (MAX_SURFS exceded)
- */
-int GS_new_surface(void)
-{
-    geosurf *ns;
-
-    G_debug(3, "GS_new_surface():");
-
-    if (Next_surf < MAX_SURFS) {
-	ns = gs_get_new_surface();
-	gs_init_surf(ns, wind.west + wind.ew_res / 2.,
-		     wind.south + wind.ns_res / 2., wind.rows, wind.cols,
-		     wind.ew_res, wind.ns_res);
-	gs_set_defaults(ns, Default_const, Default_nulls);
-
-	/* make default shine current */
-	gs_set_att_src(ns, ATT_SHINE, CONST_ATT);
-
-	Surf_ID[Next_surf] = ns->gsurf_id;
-	++Next_surf;
-
-	G_debug(3, "    id=%d", ns->gsurf_id);
-
-	return (ns->gsurf_id);
-    }
-
-
-
-    return (-1);
-}
-void GS_set_light_reset(int i)
-{
-    Resetlight = i;
-    if (i)
-	Numlights = 0;
-}
-int GS_get_light_reset(void)
-{
-    return Resetlight;
-}
-/*!
-   \brief Add new model light
-
-   \return light model id
-   \return -1 on error (MAX_LIGHTS exceded)
- */
-int GS_new_light(void)
-{
-    int i;
-
-    if (GS_get_light_reset()) {
-
-	GS_set_light_reset(0);
-
-	for (i = 0; i < MAX_LIGHTS; i++) {
-	    Gv.lights[i].position[X] = Gv.lights[i].position[Y] = 0.0;
-	    Gv.lights[i].position[Z] = 1.0;
-	    Gv.lights[i].position[W] = 0.0;	/* infinite */
-	    Gv.lights[i].color[0] = Gv.lights[i].color[1] =
-		Gv.lights[i].color[2] = 1.0;
-	    Gv.lights[i].ambient[0] = Gv.lights[i].ambient[1] =
-		Gv.lights[i].ambient[2] = 0.2;
-	    Gv.lights[i].shine = 32.0;
-	}
-
-	gsd_init_lightmodel();
-    }
-
-    if (Numlights < MAX_LIGHTS) {
-	gsd_deflight(Numlights + 1, &(Gv.lights[Numlights]));
-	gsd_switchlight(Numlights + 1, 1);
-
-	return ++Numlights;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Set light position
-
-   \bug I think lights array doesnt match sgi_light array
-
-   \param num light id (starts with 1)
-   \param xpos,ypos,zpos coordinates (model)
-   \param local local coordinate (for viewport)
- */
-void GS_setlight_position(int num, float xpos, float ypos, float zpos,
-			  int local)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    Gv.lights[num].position[X] = xpos;
-	    Gv.lights[num].position[Y] = ypos;
-	    Gv.lights[num].position[Z] = zpos;
-	    Gv.lights[num].position[W] = (float)local;
-
-	    gsd_deflight(num + 1, &(Gv.lights[num]));
-	}
-    }
-
-    return;
-}
-
-
-/*!
-   \brief Get light position
-
-   \param num light id (starts at 1)
-   \param[out] xpos,ypos,zpos coordinates
-   \param[out] local ?
- */
-void GS_getlight_position(int num, float *xpos, float *ypos, float *zpos,
-			  int *local)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    *xpos = Gv.lights[num].position[X];
-	    *ypos = Gv.lights[num].position[Y];
-	    *zpos = Gv.lights[num].position[Z];
-	    *local = (int)Gv.lights[num].position[W];
-
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Set light color
-
-   \param num light id (starts at 1)
-   \param red,green,blue color values (from 0.0 to 1.0)
- */
-void GS_setlight_color(int num, float red, float green, float blue)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    Gv.lights[num].color[0] = red;
-	    Gv.lights[num].color[1] = green;
-	    Gv.lights[num].color[2] = blue;
-
-	    gsd_deflight(num + 1, &(Gv.lights[num]));
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Get light color
-
-   \param num light id (starts at 1)
-   \param[out] red,green,blue color values
- */
-void GS_getlight_color(int num, float *red, float *green, float *blue)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    *red = Gv.lights[num].color[0];
-	    *green = Gv.lights[num].color[1];
-	    *blue = Gv.lights[num].color[2];
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Set light ambient
-
-   Red, green, blue from 0.0 to 1.0
-
-   \param num light id (starts at 1)
-   \param red,green,blue color values
- */
-void GS_setlight_ambient(int num, float red, float green, float blue)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    Gv.lights[num].ambient[0] = red;
-	    Gv.lights[num].ambient[1] = green;
-	    Gv.lights[num].ambient[2] = blue;
-
-	    gsd_deflight(num + 1, &(Gv.lights[num]));
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Get light ambient
-
-   \param num light id (starts at 1)
-   \param[out] red,green,blue color values
- */
-void GS_getlight_ambient(int num, float *red, float *green, float *blue)
-{
-    if (num) {
-	num -= 1;
-	if (num < Numlights) {
-	    *red = Gv.lights[num].ambient[0];
-	    *green = Gv.lights[num].ambient[1];
-	    *blue = Gv.lights[num].ambient[2];
-	}
-    }
-
-    return;
-}
-
-
-/*!
-   \brief Switch off all lights
- */
-void GS_lights_off(void)
-{
-    int i;
-
-    for (i = 0; i < Numlights; i++) {
-	gsd_switchlight(i + 1, 0);
-    }
-
-    return;
-}
-
-/*!
-   \brief Switch on all lights
- */
-void GS_lights_on(void)
-{
-    int i;
-
-    for (i = 0; i < Numlights; i++) {
-	gsd_switchlight(i + 1, 1);
-    }
-
-    return;
-}
-
-/*!
-   \brief Switch on/off light
-
-   \param num light id (starts at 1)
-   \param on non-zero for 'on' otherwise 'off'
- */
-void GS_switchlight(int num, int on)
-{
-    if (num) {
-	num -= 1;
-
-	if (num < Numlights) {
-	    gsd_switchlight(num + 1, on);
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Check if transparency is set
-
-   \return 0 transparency not set
-   \return 1 transparency is set
- */
-int GS_transp_is_set(void)
-{
-    return (gs_att_is_set(NULL, ATT_TRANSP) || (FC_GREY == gsd_getfc()));
-}
-
-/*!
-   \brief Retrieves coordinates for lighting model position, at center of view
-
-   \param pos[out] coordinates
- */
-void GS_get_modelposition1(float pos[])
-{
-    /* TODO: Still needs work to handle other cases */
-    /* this is a quick hack to get lighting adjustments debugged */
-    /*
-       GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], center);
-       GS_v3mult(center, 1000);
-       GS_v3add(center, Gv.from_to[FROM]);
-     */
-
-    gs_get_datacenter(pos);
-    gs_get_data_avg_zmax(&(pos[Z]));
-
-    G_debug(1, "GS_get_modelposition1(): model position: %f %f %f",
-	    pos[X], pos[Y], pos[Z]);
-
-    return;
-}
-
-/*!
-   \brief Retrieves coordinates for lighting model position, at center of view
-
-   Position at nearclip * 2: tried nearclip + siz, but since need to
-   know position to calculate size, have two dependent variables
-   (nearclip * 2) from eye.
-
-   \param siz[out] size
-   \param pos[out] coordinates (X, Y, Z)
- */
-void GS_get_modelposition(float *siz, float *pos)
-{
-    float dist, near_h, dir[3];
-
-    dist = 2. * Gd.nearclip;
-
-    near_h = 2.0 * tan(4.0 * atan(1.) * Gv.fov / 3600.) * dist;
-    *siz = near_h / 8.0;
-
-    /* prevent clipping - would only happen if fov > ~127 degrees, at
-       fov = 2.0 * atan(2.0) */
-
-    if (*siz > Gd.nearclip) {
-	*siz = Gd.nearclip;
-    }
-
-    GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], dir);
-
-    pos[X] = Gv.from_to[FROM][X] + dir[X] * dist;
-    pos[Y] = Gv.from_to[FROM][Y] + dir[Y] * dist;
-    pos[Z] = Gv.from_to[FROM][Z] + dir[Z] * dist;
-
-    return;
-}
-
-
-/*!
-   \brief Set decoration, north arrow ??
-
-   \todo scale used to calculate len of arrow still needs work
-   needs go function that returns center / eye distance
-   gsd_get_los function is not working correctly ??
-
-   \param pt point value in true world coordinates (?)
-   \param id surface id
-   \param[out] pos2 output coordinates
- */
-void GS_set_Narrow(int *pt, int id, float *pos2)
-{
-    geosurf *gs;
-    float x, y, z;
-    GLdouble modelMatrix[16], projMatrix[16];
-    GLint viewport[4];
-
-    if (GS_get_selected_point_on_surface(pt[X], pt[Y], &id, &x, &y, &z)) {
-	gs = gs_get_surf(id);
-	if (gs) {
-	    z = gs->zmax;
-	    pos2[X] = (float)x - gs->ox + gs->x_trans;
-	    pos2[Y] = (float)y - gs->oy + gs->y_trans;
-	    pos2[Z] = (float)z + gs->z_trans;
-
-	    return;
-	}
-    }
-    else {
-	gs = gs_get_surf(id);
-
-	/* Need to get model matrix, etc 
-	 * to run gluUnProject
-	 */
-	gsd_pushmatrix();
-	gsd_do_scale(1);
-	glGetDoublev(GL_MODELVIEW_MATRIX, modelMatrix);
-	glGetDoublev(GL_PROJECTION_MATRIX, projMatrix);
-	glGetIntegerv(GL_VIEWPORT, viewport);
-
-	if (gs) {
-	    GLdouble out_near[3], out_far[3];
-	    GLdouble factor;
-	    GLdouble out[3];
-
-	    z = (float)gs->zmax + gs->z_trans;
-
-	    gluUnProject((GLdouble) pt[X], (GLdouble) pt[Y], (GLdouble) 0.,
-			 modelMatrix, projMatrix, viewport,
-			 &out_near[X], &out_near[Y], &out_near[Z]);
-	    gluUnProject((GLdouble) pt[X], (GLdouble) pt[Y], (GLdouble) 1.,
-			 modelMatrix, projMatrix, viewport,
-			 &out_far[X], &out_far[Y], &out_far[Z]);
-
-	    glPopMatrix();
-
-	    factor = (out_near[Z] - z) / (out_near[Z] - out_far[Z]);
-
-	    out[X] = out_near[X] - ((out_near[X] - out_far[X]) * factor);
-	    out[Y] = out_near[Y] - ((out_near[Y] - out_far[Y]) * factor);
-	    out[Z] = z;
-
-	    pos2[X] = (float)out[X];
-	    pos2[Y] = (float)out[Y];
-	    pos2[Z] = (float)out[Z];
-
-	    return;
-
-	}
-    }
-    return;
-}
-
-/*!
-   \brief Draw place marker
-
-   Used to display query point for raster queries.
-
-   \param id surface id
-   \param pt point, X, Y value in true world coordinates
- */
-void GS_draw_X(int id, float *pt)
-{
-    geosurf *gs;
-    Point3 pos;
-    float siz;
-    gvstyle style;
-
-    if ((gs = gs_get_surf(id))) {
-	GS_get_longdim(&siz);
-	style.size = siz / 200.;
-	pos[X] = pt[X] - gs->ox;
-	pos[Y] = pt[Y] - gs->oy;
-	_viewcell_tri_interp(gs, pos);
-
-	gsd_pushmatrix();
-
-	gsd_do_scale(1);
-	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
-	gsd_linewidth(1);
-
-	if (CONST_ATT == gs_get_att_src(gs, ATT_TOPO)) {
-	    pos[Z] = gs->att[ATT_TOPO].constant;
-	    gs = NULL;		/* tells gpd_obj to use given Z val */
-	}
-	style.color = Gd.bgcol;
-	style.symbol = ST_GYRO;
-	gpd_obj(gs, &style, pos);
-	gsd_flush();
-
-	gsd_popmatrix();
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw line on surface
-
-   \param id surface id
-   \param x1,y1,x2,y2 line nodes
- */
-void GS_draw_line_onsurf(int id, float x1, float y1, float x2, float y2)
-{
-    float p1[2], p2[2];
-    geosurf *gs;
-
-    if ((gs = gs_get_surf(id))) {
-	p1[X] = x1 - gs->ox;
-	p1[Y] = y1 - gs->oy;
-	p2[X] = x2 - gs->ox;
-	p2[Y] = y2 - gs->oy;
-
-	gsd_pushmatrix();
-
-	gsd_do_scale(1);
-	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
-	gsd_linewidth(1);
-
-	gsd_color_func(GS_default_draw_color());
-	gsd_line_onsurf(gs, p1, p2);
-
-	gsd_popmatrix();
-	gsd_flush();
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw multiline on surface
-
-   Like above but limits points in line to n or points found in segment,
-   whichever is smaller.
-
-   \param id surface id
-   \param x1,y1,x2,y2 line nodes
-
-   \return number of points used
- */
-int GS_draw_nline_onsurf(int id, float x1, float y1, float x2, float y2,
-			 float *lasp, int n)
-{
-    float p1[2], p2[2];
-    geosurf *gs;
-    int ret = 0;
-
-    if ((gs = gs_get_surf(id))) {
-	p1[X] = x1 - gs->ox;
-	p1[Y] = y1 - gs->oy;
-	p2[X] = x2 - gs->ox;
-	p2[Y] = y2 - gs->oy;
-
-	gsd_pushmatrix();
-
-	gsd_do_scale(1);
-	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
-	gsd_linewidth(1);
-	gsd_color_func(GS_default_draw_color());
-	ret = gsd_nline_onsurf(gs, p1, p2, lasp, n);
-	gsd_surf2real(gs, lasp);
-
-	gsd_popmatrix();
-	gsd_flush();
-    }
-
-    return (ret);
-}
-
-/*!
-   \brief Draw flow-line on surace
-
-   This is slow - should be moved to gs_ but GS_ good for testing
-   and useful for app programmer
-
-   \param id surface id
-   \param x,y coordinates of flow-line
- */
-void GS_draw_flowline_at_xy(int id, float x, float y)
-{
-    geosurf *gs;
-    float nv[3], pdir[2], mult;
-    float p1[2], p2[2], next[2];
-    int i = 0;
-
-    if ((gs = gs_get_surf(id))) {
-	p1[X] = x;
-	p1[Y] = y;
-	/* multiply by 1.5 resolutions to ensure a crossing ? */
-	mult = .1 * (VXRES(gs) > VYRES(gs) ? VXRES(gs) : VYRES(gs));
-
-	GS_coordpair_repeats(p1, p1, 50);
-
-	while (1 == GS_get_norm_at_xy(id, p1[X], p1[Y], nv)) {
-	    if (nv[Z] == 1.0) {
-		if (pdir[X] == 0.0 && pdir[Y] == 0.0) {
-		    break;
-		}
-
-		p2[X] = p1[X] + (pdir[X] * mult);
-		p2[Y] = p1[Y] + (pdir[Y] * mult);
-	    }
-	    else {
-		/* use previous direction */
-		GS_v2norm(nv);
-		p2[X] = p1[X] + (nv[X] * mult);
-		p2[Y] = p1[Y] + (nv[Y] * mult);
-		pdir[X] = nv[X];
-		pdir[Y] = nv[Y];
-	    }
-
-	    if (i > 2000) {
-		break;
-	    }
-
-	    if (GS_coordpair_repeats(p1, p2, 0)) {
-		break;
-	    }
-
-	    /* Think about this: */
-	    /* degenerate line means edge or level edge ? */
-	    /* next is filled with last point drawn */
-	    if (2 > GS_draw_nline_onsurf(id, p1[X], p1[Y],
-					 p2[X], p2[Y], next, 3)) {
-		break;
-	    }
-
-	    p1[X] = next[X];
-	    p1[Y] = next[Y];
-	}
-
-	G_debug(3, "GS_draw_flowline_at_xy(): dir: %f %f", nv[X], nv[Y]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw fringe around data (surface) at selected corners
-
-   \param id surface id
-   \param clr color
-   \param elev elevation value
-   \param where nw/ne/sw/se edges - 0 (turn off) 1 (turn on)
- */
-void GS_draw_fringe(int id, unsigned long clr, float elev, int *where)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_draw_fringe(): id: %d clr: %ld elev %f edges: %d %d %d %d",
-	    id, clr, elev, where[0], where[1], where[2], where[3]);
-    if ((gs = gs_get_surf(id)))
-	gsd_display_fringe(gs, clr, elev, where);
-
-}
-
-
-/*!
-   \brief Draw legend
-
-   \todo add legend from list option
-   make font loading more flexible
-
-   \param name legend name
-   \param fontbase font-base
-   \param size ? 
-   \param flags legend flags
-   \param range values range
-   \param pt ?
- */
-int GS_draw_legend(const char *name, GLuint fontbase, int size, int *flags,
-		   float *range, int *pt)
-{
-    int list_no;
-
-    list_no = gsd_put_legend(name, fontbase, size, flags, range, pt);
-
-    return (list_no);
-}
-
-/*!
-   \brief Draw pre-defined list
-
-   Uses glFlush() to ensure all drawing is complete
-   before returning
-
-   \param list_id list id
- */
-void GS_draw_list(GLuint list_id)
-{
-    gsd_calllist(list_id);
-    glFlush();
-    return;
-}
-
-/*!
-   \brief Draw all glLists
-
-   Uses glFlush() to ensure all drawing is complete
-   before returning
- */
-void GS_draw_all_list(void)
-{
-    gsd_calllists(0);		/* not sure if 0 is right - MN */
-    glFlush();
-    return;
-}
-
-/*!
-   \brief Delete pre-defined list
-
-   \param list_id list id
- */
-void GS_delete_list(GLuint list_id)
-{
-    gsd_deletelist(list_id, 1);
-
-    return;
-}
-
-/*!
-   \brief Draw lighting model
- */
-void GS_draw_lighting_model1(void)
-{
-    static float center[3];
-    float tcenter[3];
-
-    if (!Modelshowing) {
-	GS_get_modelposition1(center);
-    }
-
-    GS_v3eq(tcenter, center);
-
-    gsd_zwritemask(0x0);
-    gsd_backface(1);
-
-    gsd_colormode(CM_AD);
-    gsd_shademodel(DM_GOURAUD);
-    gsd_pushmatrix();
-    gsd_do_scale(1);
-
-    if (Gv.vert_exag) {
-	tcenter[Z] *= Gv.vert_exag;
-	gsd_scale(1.0, 1.0, 1. / Gv.vert_exag);
-    }
-
-    gsd_drawsphere(tcenter, 0xDDDDDD, (float)(Longdim / 10.));
-    gsd_popmatrix();
-    Modelshowing = 1;
-
-    gsd_backface(0);
-    gsd_zwritemask(0xffffffff);
-
-    return;
-}
-
-/*!
-   \brief Draw lighting model
-
-   Just turn off any cutting planes and draw it just outside near
-   clipping plane, since lighting is infinite now
- */
-void GS_draw_lighting_model(void)
-{
-    static float center[3], size;
-    float tcenter[3], tsize;
-    int i, wason[MAX_CPLANES];
-
-    gsd_get_cplanes_state(wason);
-
-    for (i = 0; i < MAX_CPLANES; i++) {
-	if (wason[i]) {
-	    gsd_cplane_off(i);
-	}
-    }
-
-
-    if (!Modelshowing) {
-	GS_get_modelposition(&size, center);
-    }
-
-    GS_v3eq(tcenter, center);
-    tsize = size;
-
-    gsd_zwritemask(0x0);
-    gsd_backface(1);
-
-    gsd_colormode(CM_DIFFUSE);
-    gsd_shademodel(DM_GOURAUD);
-    gsd_pushmatrix();
-    gsd_drawsphere(tcenter, 0xDDDDDD, tsize);
-    gsd_popmatrix();
-    Modelshowing = 1;
-
-    gsd_backface(0);
-    gsd_zwritemask(0xffffffff);
-
-    for (i = 0; i < MAX_CPLANES; i++) {
-	if (wason[i]) {
-	    gsd_cplane_on(i);
-	}
-    }
-
-    gsd_flush();
-
-    return;
-}
-
-/*!
-   \brief Update current mask
-
-   May be called to update total mask for a surface at convenient times
-   instead of waiting until ready to redraw surface
-
-   \param id surface id
-
-   \return ?
- */
-int GS_update_curmask(int id)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-    return (gs_update_curmask(gs));
-}
-
-/*!
-   \brief Check if point is masked ?
-
-   \param id surface id
-   \param pt point
-
-   \return 1 masked
-   \return 0 not masked
-   \return -1 on error, invalid surface id
- */
-int GS_is_masked(int id, float *pt)
-{
-    geosurf *gs;
-    Point3 tmp;
-
-    if ((gs = gs_get_surf(id))) {
-	tmp[X] = pt[X] - gs->ox;
-	tmp[Y] = pt[Y] - gs->oy;
-
-	return (gs_point_is_masked(gs, tmp));
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Unset Scaled Difference surface
- */
-void GS_unset_SDsurf(void)
-{
-    gsdiff_set_SDref(NULL);
-    SDref_surf = 0;
-
-    return;
-}
-
-/*!
-   \brief Set surface as Scaled Difference surface
-
-   \param id surface id
-
-   \return 1 on success
-   \return 0 on error, invalid surface id
- */
-int GS_set_SDsurf(int id)
-{
-    geosurf *gs;
-
-    if ((gs = gs_get_surf(id))) {
-	gsdiff_set_SDref(gs);
-	SDref_surf = id;
-
-	return (1);
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Set ?
-
-   \param scale scale value
-
-   \return 1
- */
-int GS_set_SDscale(float scale)
-{
-    gsdiff_set_SDscale(scale);
-
-    return (1);
-}
-
-/*!
-   \brief Get ?
-
-   \param[out] id ?
-
-   \return 1 on success
-   \return 0 on error
- */
-int GS_get_SDsurf(int *id)
-{
-    geosurf *gs;
-
-    if ((gs = gsdiff_get_SDref())) {
-	*id = SDref_surf;
-
-	return (1);
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Get ?
-
-   \param[out] scale value
-
-   \return 1
- */
-int GS_get_SDscale(float *scale)
-{
-    *scale = gsdiff_get_SDscale();
-
-    return (1);
-}
-
-/*!
-   \brief Update normals
-
-   \param id surface id
-
-   \return ?
- */
-int GS_update_normals(int id)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    return (gs_calc_normals(gs));
-}
-
-/*!
-   \brief Get attributes
-
-   \param id surface id
-   \param att
-   \param[out] set
-   \param[out] constant
-   \param[out] mapname
-
-   \return 1 on success
-   \return -1 on error (invalid surface id)
- */
-int GS_get_att(int id, int att, int *set, float *constant, char *mapname)
-{
-    int src;
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-    if (gs) {
-	if (-1 != (src = gs_get_att_src(gs, att))) {
-	    *set = src;
-
-	    if (src == CONST_ATT) {
-		*constant = gs->att[att].constant;
-	    }
-	    else if (src == MAP_ATT) {
-		strcpy(mapname, gsds_get_name(gs->att[att].hdata));
-	    }
-
-	    return (1);
-	}
-
-	return (-1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get surface category on given position
-
-   Prints "no data" or a description (i.e., "coniferous forest") to
-   <i>catstr</i>. Usually call after GS_get_selected_point_on_surface().
-   Define <i>att</i> as MAP_ATT
-
-   \todo Allocate catstr using G_store()
-   
-   \param id surface id
-   \param att attribute id (MAP_ATT)
-   \param catstr cat string (must be allocated, dim?)
-   \param x,y real coordinates
-
-   \return -1 if no category info or point outside of window
-   \return 1 on success
-*/
-int GS_get_cat_at_xy(int id, int att, char *catstr, float x, float y)
-{
-    int offset, drow, dcol, vrow, vcol;
-    float ftmp, pt[3];
-    typbuff *buff;
-    geosurf *gs;
-
-    *catstr = '\0';
-    gs = gs_get_surf(id);
-
-    if (NULL == gs) {
-	return -1;
-    }
-
-    pt[X] = x;
-    pt[Y] = y;
-
-    gsd_real2surf(gs, pt);
-    if (gs_point_is_masked(gs, pt)) {
-	return -1;
-    }
-
-    if (!in_vregion(gs, pt)) {
-	return -1;
-    }
-
-    if (MAP_ATT != gs_get_att_src(gs, att)) {
-	sprintf(catstr, _("no category info"));
-	return -1;
-    }
-
-    buff = gs_get_att_typbuff(gs, att, 0);
-
-    vrow = Y2VROW(gs, pt[Y]);
-    vcol = X2VCOL(gs, pt[X]);
-    drow = VROW2DROW(gs, vrow);
-    dcol = VCOL2DCOL(gs, vcol);
-
-    offset = DRC2OFF(gs, drow, dcol);
-    
-    if (GET_MAPATT(buff, offset, ftmp)) {
-	return
-	    (Gs_get_cat_label(gsds_get_name(gs->att[att].hdata),
-			      drow, dcol, catstr));
-    }
-
-    sprintf(catstr, _("no data"));
-
-    return 1;
-}
-
-/*!
-   \brief Get surface normal at x,y (real coordinates)
-
-   Usually call after GS_get_selected_point_on_surface()
-
-   \param id surface id
-   \param x,y real coordinates
-   \param[out] nv surface normal
-
-   \return -1 if point outside of window or masked
-   \return 1 on success
- */
-int GS_get_norm_at_xy(int id, float x, float y, float *nv)
-{
-    int offset, drow, dcol, vrow, vcol;
-    float pt[3];
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (NULL == gs) {
-	return (-1);
-    }
-
-    if (gs->norm_needupdate) {
-	gs_calc_normals(gs);
-    }
-
-    pt[X] = x;
-    pt[Y] = y;
-
-    gsd_real2surf(gs, pt);
-    if (gs_point_is_masked(gs, pt)) {
-	return (-1);
-    }
-
-    if (!in_vregion(gs, pt)) {
-	return (-1);
-    }
-
-    vrow = Y2VROW(gs, pt[Y]);
-    vcol = X2VCOL(gs, pt[X]);
-    drow = VROW2DROW(gs, vrow);
-    dcol = VCOL2DCOL(gs, vcol);
-
-    offset = DRC2OFF(gs, drow, dcol);
-
-    if (gs->norms) {
-	FNORM(gs->norms[offset], nv);
-    }
-    else {
-	/* otherwise must be a constant */
-	nv[0] = 0.0;
-	nv[1] = 0.0;
-	nv[2] = 1.0;
-    }
-
-    return (1);
-}
-
-/*!
-   \brief Get RGB color at given point
-
-   Colors are translated to rgb and returned as Rxxx Gxxx Bxxx Usually
-   call after GS_get_selected_point_on_surface().
-
-   Prints NULL or the value (i.e., "921.5") to valstr
-
-   \param id surface id
-   \param att attribute id
-   \param[out] valstr value string (allocated, dim?)
-   \param x,y real coordinates
-   
-   \return -1 if point outside of window or masked
-   \return 1 on success
- */
-int GS_get_val_at_xy(int id, int att, char *valstr, float x, float y)
-{
-    int offset, drow, dcol, vrow, vcol;
-    float ftmp, pt[3];
-    typbuff *buff;
-    geosurf *gs;
-
-    *valstr = '\0';
-    gs = gs_get_surf(id);
-    
-    if (NULL == gs) {
-	return -1;
-    }
-
-    pt[X] = x;
-    pt[Y] = y;
-
-    gsd_real2surf(gs, pt);
-
-    if (gs_point_is_masked(gs, pt)) {
-	return -1;
-    }
-
-    if (!in_vregion(gs, pt)) {
-	return (-1);
-    }
-
-    if (CONST_ATT == gs_get_att_src(gs, att)) {
-	if (att == ATT_COLOR) {
-	    int r, g, b, i;
-
-	    i = gs->att[att].constant;
-	    sprintf(valstr, "R%d G%d B%d",
-		    INT_TO_RED(i, r), INT_TO_GRN(i, g), INT_TO_BLU(i, b));
-	}
-	else {
-	    sprintf(valstr, "%f", gs->att[att].constant);
-	}
-
-	return 1;
-    }
-    else if (MAP_ATT != gs_get_att_src(gs, att)) {
-	return -1;
-    }
-
-    buff = gs_get_att_typbuff(gs, att, 0);
-
-    vrow = Y2VROW(gs, pt[Y]);
-    vcol = X2VCOL(gs, pt[X]);
-    drow = VROW2DROW(gs, vrow);
-    dcol = VCOL2DCOL(gs, vcol);
-
-    offset = DRC2OFF(gs, drow, dcol);
-
-    if (GET_MAPATT(buff, offset, ftmp)) {
-	if (att == ATT_COLOR) {
-	    int r, g, b, i;
-
-	    i = gs_mapcolor(gs_get_att_typbuff(gs, ATT_COLOR, 0),
-			    &(gs->att[ATT_COLOR]), offset);
-	    sprintf(valstr, "R%d G%d B%d",
-		    INT_TO_RED(i, r), INT_TO_GRN(i, g), INT_TO_BLU(i, b));
-	}
-	else {
-	    sprintf(valstr, "%f", ftmp);
-	}
-
-	return (1);
-    }
-
-    sprintf(valstr, "NULL");
-
-    return (1);
-}
-
-/*!
-   \brief Unset attribute
-
-   \param id surface id
-   \param att attribute id
-
-   \return ?
- */
-int GS_unset_att(int id, int att)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-    gs->mask_needupdate = 1;
-
-    return (gs_set_att_src(gs, att, NOTSET_ATT));
-}
-
-/*!
-   \brief Set attribute constant
-
-   \param id surface id
-   \param att attribute id
-   \param constant value
-
-   \return ?
- */
-int GS_set_att_const(int id, int att, float constant)
-{
-    geosurf *gs;
-    int ret;
-
-    gs = gs_get_surf(id);
-    ret = (gs_set_att_const(gs, att, constant));
-
-    Gs_update_attrange(gs, att);
-
-    return (ret);
-}
-
-/*!
-   \brief Set mask mode
-
-   Mask attribute special: constant is set to indicate invert or no
-
-   \param id surface id
-   \param mode id
-
-   \return mode id
-   \return -1 on error (invalid surface id)
- */
-int GS_set_maskmode(int id, int mode)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	gs->att[ATT_MASK].constant = mode;
-	gs->mask_needupdate = 1;
-
-	return (mode);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get mask mode
-
-   \param id surface id
-   \param[out] mode id
-
-   \return 1 on success
-   \return -1 on error (invalid surface id)
- */
-int GS_get_maskmode(int id, int *mode)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*mode = gs->att[ATT_MASK].constant;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set client data
-
-   \param id surface id
-   \param clientd pointer to client data struct
-
-   \return 1 on success
-   \return -1 on error (invalid surface id)
- */
-int GS_Set_ClientData(int id, void *clientd)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-    if (gs) {
-	gs->clientdata = clientd;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get client data
-
-   \param id surface id
-
-   \return pointer to client data
-   \return NULL on error
- */
-void *GS_Get_ClientData(int id)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-    if (gs) {
-	return (gs->clientdata);
-    }
-
-    return (NULL);
-}
-
-/*!
-   \brief Get number of surfaces
-
-   \return number of surfaces
- */
-int GS_num_surfs(void)
-{
-    return (gs_num_surfaces());
-}
-
-/*!
-   \brief Get surface list
-
-   Must be freed when not neeed!
-
-   \param[out] numsurf number of available surfaces
-
-   \return pointer to surface array
-   \return NULL on error
- */
-int *GS_get_surf_list(int *numsurfs)
-{
-    int i, *ret;
-
-    *numsurfs = Next_surf;
-
-    if (Next_surf) {
-	ret = (int *)G_malloc(Next_surf * sizeof(int));
-
-	for (i = 0; i < Next_surf; i++) {
-	    ret[i] = Surf_ID[i];
-	}
-
-	return (ret);
-    }
-
-    return (NULL);
-}
-
-/*!
-   \brief Delete surface
-
-   \param id surface id
-
-   \return 1 on success
-   \return -1 on error
- */
-int GS_delete_surface(int id)
-{
-    int i, j, found;
-    
-    found = FALSE;
-    
-    G_debug(1, "GS_delete_surface(): id=%d", id);
-    
-    if (GS_surf_exists(id)) {
-	gs_delete_surf(id);
-	for (i = 0; i < Next_surf && !found; i++) {
-	    if (Surf_ID[i] == id) {
-		found = TRUE;
-
-		for (j = i; j < Next_surf; j++) {
-		    Surf_ID[j] = Surf_ID[j + 1];
-		}
-	    }
-	}
-	
-	gv_update_drapesurfs();
-
-	if (found) {
-	    --Next_surf;
-	    return 1;
-	}
-    }
-
-    return -1;
-}
-
-
-/*!
-   \brief Load raster map as attribute
-
-   \param id surface id
-   \param filename filename
-   \param att attribute descriptor
-
-   \return -1 on error (invalid surface id)
-   \return ?
- */
-int GS_load_att_map(int id, const char *filename, int att)
-{
-    geosurf *gs;
-    unsigned int changed;
-    unsigned int atty;
-    const char *mapset;
-    struct Cell_head rast_head;
-    int reuse, begin, hdata, ret, neg, has_null;
-    typbuff *tbuff;
-
-    G_debug(3, "GS_load_att_map(): map=%s", filename);
-
-    reuse = ret = neg = has_null = 0;
-    gs = gs_get_surf(id);
-
-    if (NULL == gs) {
-	return -1;
-    }
-
-    gs->mask_needupdate = (ATT_MASK == att || ATT_TOPO == att ||
-			   (gs->nz_topo && ATT_TOPO == att) ||
-			   (gs->nz_color && ATT_COLOR == att));
-
-    gs_set_att_src(gs, att, MAP_ATT);
-
-    /* Check against maps already loaded in memory   */
-    /* if to be color attribute:
-       - if packed color for another surface, OK to reuse
-       - if unchanged, ok to reuse IF it's of type char (will have lookup)
-     */
-    begin = hdata = 1;
-
-    /* Get MAPSET to ensure names are fully qualified */
-    mapset = G_find_raster2(filename, "");
-    if (mapset == NULL) {
-	/* Check for valid filename */
-	G_warning("Raster map <%s> not found", filename);
-	return -1;
-    }
-    
-    /* Check to see if map is in Region */
-    Rast_get_cellhd(filename, mapset, &rast_head);
-    if (rast_head.north <= wind.south ||
-	rast_head.south >= wind.north ||
-	rast_head.east <= wind.west || rast_head.west >= wind.east) {
-
-	G_warning(_("Raster map <%s> is outside of current region. Load failed."),
-		  G_fully_qualified_name(filename, mapset));
-    }
-
-    while (!reuse && (0 < hdata)) {
-	changed = CF_COLOR_PACKED;
-	atty = ATTY_FLOAT | ATTY_CHAR | ATTY_INT | ATTY_SHORT | ATTY_MASK;
-
-	if (0 < (hdata = gsds_findh(filename, &changed, &atty, begin))) {
-
-	    G_debug(3, "GS_load_att_map(): %s already has data handle %d.CF=%x",
-		    filename, hdata, changed);
-
-	    /* handle found */
-	    if (ATT_COLOR == att) {
-		if ((changed == CF_COLOR_PACKED) ||
-		    (!changed && atty == ATTY_CHAR)) {
-		    reuse = 1;
-		}
-	    }
-	    else if (atty == ATTY_MASK && att != ATT_MASK) {
-		reuse = 0;
-		/* should also free mask data & share new - but need backward
-		   reference? */
-	    }
-	    else if (!changed) {
-		reuse = 1;
-	    }
-	}
-
-	begin = 0;
-    }
-
-    if (reuse) {
-	gs->att[att].hdata = hdata;
-	gs_set_att_type(gs, att, atty);	/* ?? */
-
-	/* free lookup  & set to NULL! */
-	if (atty == ATTY_INT) {
-	    if (gs->att[att].lookup) {
-		free(gs->att[att].lookup);
-		gs->att[att].lookup = NULL;
-	    }
-	}
-	/* TODO: FIX THIS stuff with lookup sharing! */
-
-	G_debug(3, "GS_load_att_map(): %s is being reused. hdata=%d",
-		filename, hdata);
-    }
-    else {
-	G_debug(3, "GS_load_att_map(): %s not loaded in correct form - loading now",
-		filename);
-
-	/* not loaded - need to get new dataset handle */
-	gs->att[att].hdata = gsds_newh(filename);
-
-	tbuff = gs_get_att_typbuff(gs, att, 1);
-
-	/* TODO: Provide mechanism for loading certain attributes at
-	   specified sizes, allow to scale or cap, or scale non-zero */
-	if (ATT_MASK == att) {
-	    atty = ATTY_MASK;
-	}
-	else {
-	    atty = Gs_numtype(filename, &neg);
-	}
-
-#ifdef MAYBE_LATER
-	if (att == ATT_COLOR && atty == ATTY_SHORT) {
-	    atty = (neg ? ATTY_INT : ATTY_SHORT);
-	}
-#endif
-
-	if (att == ATT_COLOR && atty == ATTY_SHORT) {
-	    atty = ATTY_INT;
-	}
-
-	if (0 == gs_malloc_att_buff(gs, att, ATTY_NULL)) {
-	    G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	}
-
-	switch (atty) {
-	case ATTY_MASK:
-	    if (0 == gs_malloc_att_buff(gs, att, ATTY_MASK)) {
-		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	    }
-
-	    ret = Gs_loadmap_as_bitmap(&wind, filename, tbuff->bm);
-	    
-	    break;
-	case ATTY_CHAR:
-	    if (0 == gs_malloc_att_buff(gs, att, ATTY_CHAR)) {
-		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	    }
-
-	    ret = Gs_loadmap_as_char(&wind, filename, tbuff->cb,
-				     tbuff->nm, &has_null);
-
-	    break;
-	case ATTY_SHORT:
-	    if (0 == gs_malloc_att_buff(gs, att, ATTY_SHORT)) {
-		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	    }
-
-	    ret = Gs_loadmap_as_short(&wind, filename, tbuff->sb,
-				      tbuff->nm, &has_null);
-	    break;
-	case ATTY_FLOAT:
-	    if (0 == gs_malloc_att_buff(gs, att, ATTY_FLOAT)) {
-		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	    }
-
-	    ret = Gs_loadmap_as_float(&wind, filename, tbuff->fb,
-				      tbuff->nm, &has_null);
-
-	    break;
-	case ATTY_INT:
-	default:
-	    if (0 == gs_malloc_att_buff(gs, att, ATTY_INT)) {
-		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-	    }
-
-	    ret = Gs_loadmap_as_int(&wind, filename, tbuff->ib,
-				    tbuff->nm, &has_null);
-	    break;
-
-	}			/* Done with switch */
-
-	if (ret == -1) {
-	    gsds_free_data_buff(gs->att[att].hdata, ATTY_NULL);
-	    return -1;
-	}
-
-	G_debug(4, "  has_null=%d", has_null);
-
-	if (!has_null) {
-	    gsds_free_data_buff(gs->att[att].hdata, ATTY_NULL);
-	}
-	else {
-	    gs_update_curmask(gs);
-	}
-
-    }				/* end if not reuse */
-
-    if (ATT_COLOR == att) {
-#ifdef MAYBE_LATER
-	if (ATTY_INT == atty) {
-	    Gs_pack_colors(filename, tbuff->ib, gs->rows, gs->cols);
-	    gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
-	    gs->att[att].lookup = NULL;
-	}
-	else {
-	    gs_malloc_lookup(gs, att);
-	    Gs_build_lookup(filename, gs->att[att].lookup);
-	}
-#else
-
-	if (ATTY_CHAR == atty) {
-	    if (!gs->att[att].lookup) {
-		/* might already exist if reusing */
-		gs_malloc_lookup(gs, att);
-		Gs_build_256lookup(filename, gs->att[att].lookup);
-	    }
-	}
-	else if (ATTY_FLOAT == atty) {
-	    if (!reuse) {
-		if (0 == gs_malloc_att_buff(gs, att, ATTY_INT)) {
-		    G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
-		}
-
-		Gs_pack_colors_float(filename, tbuff->fb, tbuff->ib,
-				     gs->rows, gs->cols);
-		gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
-		gsds_free_data_buff(gs->att[att].hdata, ATTY_FLOAT);
-		gs->att[att].lookup = NULL;
-	    }
-	}
-	else {
-	    if (!reuse) {
-		Gs_pack_colors(filename, tbuff->ib, gs->rows, gs->cols);
-		gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
-		gs->att[att].lookup = NULL;
-	    }
-	}
-#endif
-    }
-
-    if (ATT_TOPO == att) {
-	gs_init_normbuff(gs);
-	/* S_DIFF: should also check here to see if this surface is a
-	   reference surface for scaled differences, if so update references
-	   to it */
-    }
-
-    if (ret < 0) {
-	G_warning(_("Loading failed"));
-    }
-
-    if (-1 == Gs_update_attrange(gs, att)) {
-	G_warning(_("Error finding range"));
-    }
-
-    return ret;
-}
-
-/*!
-   \brief Draw surface
-
-   \param id surface id
- */
-void GS_draw_surf(int id)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_draw_surf(): id=%d", id);
-
-    gs = gs_get_surf(id);
-    if (gs) {
-	gsd_shademodel(gs->draw_mode & DM_GOURAUD);
-
-	if (gs->draw_mode & DM_POLY) {
-	    gsd_surf(gs);
-	}
-
-	if (gs->draw_mode & DM_WIRE) {
-	    gsd_wire_surf(gs);
-	}
-
-	/* TODO: write wire/poly draw routines */
-	if (gs->draw_mode & DM_WIRE_POLY) {
-	    gsd_surf(gs);
-	    gsd_wire_surf(gs);
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw surface wire
-
-   Overrides draw_mode for fast display
-
-   \param id surface id
- */
-void GS_draw_wire(int id)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_draw_wire(): id=%d", id);
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	gsd_wire_surf(gs);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all wires
-
-   Overrides draw_mode for fast display
- */
-void GS_alldraw_wire(void)
-{
-    geosurf *gs;
-    int i;
-
-    for (i = 0; i < Next_surf; i++) {
-	if ((gs = gs_get_surf(Surf_ID[i]))) {
-	    gsd_wire_surf(gs);
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all surfaces
- */
-void GS_alldraw_surf(void)
-{
-    int i;
-
-    for (i = 0; i < Next_surf; i++) {
-	GS_draw_surf(Surf_ID[i]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Set Z exag for surface
-
-   \param id surface id
-   \param exag z-exag value
- */
-void GS_set_exag(int id, float exag)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_exag");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	if (gs->z_exag != exag) {
-	    gs->norm_needupdate = 1;
-	}
-
-	gs->z_exag = exag;
-    }
-
-    return;
-}
-
-/*!
-   \brief Set global z-exag value
-
-   \param exag exag value to be set up
- */
-void GS_set_global_exag(float exag)
-{
-
-    G_debug(3, "GS_set_global_exag");
-
-    Gv.vert_exag = exag;
-    /* GL_NORMALIZE */
-    /* Only need to update norms gs_norms.c
-     * if exag is used in norm equation which
-     * it is not! If GL_NORMALIZE is disabled
-     * will need to include.
-     gs_setall_norm_needupdate();
-     */
-
-    return;
-}
-
-/*!
-   \brief Get global z-exag value
-
-   \return value
- */
-float GS_global_exag(void)
-{
-    G_debug(3, "GS_global_exag(): %g", Gv.vert_exag);
-
-    return (Gv.vert_exag);
-}
-
-/*!
-   \brief Set wire color
-
-   \todo error-handling
-
-   \param id surface id
-   \param colr color value
- */
-void GS_set_wire_color(int id, int colr)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_wire_color");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	gs->wire_color = colr;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get wire color
-
-   \param id surface id
-   \param[out] colr color value
-
-   \return 1 on success
-   \return -1 on error
- */
-int GS_get_wire_color(int id, int *colr)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*colr = gs->wire_color;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set all draw-modes
-
-   \param mode mode id
-
-   \return 0 on success
-   \return -1 on error
- */
-int GS_setall_drawmode(int mode)
-{
-    int i;
-
-    for (i = 0; i < Next_surf; i++) {
-	if (0 != GS_set_drawmode(Surf_ID[i], mode)) {
-	    return (-1);
-	}
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Set draw mode
-
-   \param id surface id
-   \param mode mode type(s)
-
-   \return 0 on success
-   \return -1 on error (invalid surface id)
- */
-int GS_set_drawmode(int id, int mode)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_drawmode(): id=%d mode=%d", id, mode);
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	gs->draw_mode = mode;
-
-	return (0);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get draw mode
-
-   \param id surface id
-   \param[out] mode mode id
-
-   \return 1 on success
-   \return -1 on error (invalid surface id)
- */
-int GS_get_drawmode(int id, int *mode)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*mode = gs->draw_mode;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set no-zero ?
-
-   \param id surface id
-   \param att attribute id
-   \param mode mode id
- */
-void GS_set_nozero(int id, int att, int mode)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_nozero");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	if (att == ATT_TOPO) {
-	    gs->nz_topo = mode;
-	    gs->mask_needupdate = 1;
-	}
-
-	if (att == ATT_COLOR) {
-	    gs->nz_color = mode;
-	    gs->mask_needupdate = 1;
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Get no-zero ?
-
-   \param id surface id
-   \param att attribute id
-   \param[out] mode mode id
-
-   \return -1 on error (invalid surface id)
-   \return 1 on success
- */
-int GS_get_nozero(int id, int att, int *mode)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_nozero");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	if (att == ATT_TOPO) {
-	    *mode = gs->nz_topo;
-	}
-	else if (att == ATT_COLOR) {
-	    *mode = gs->nz_color;
-	}
-	else {
-	    return (-1);
-	}
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set all draw resolutions
-
-   \param xres,yres x/y resolution value
-   \param xwire,ywire x/y wire value
-
-   \return 0 on success
-   \return -1 on error
- */
-int GS_setall_drawres(int xres, int yres, int xwire, int ywire)
-{
-    int i;
-
-    for (i = 0; i < Next_surf; i++) {
-	if (0 != GS_set_drawres(Surf_ID[i], xres, yres, xwire, ywire)) {
-	    return (-1);
-	}
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Set draw resolution for surface
-
-   \param id surface id
-   \param xres,yres x/y resolution value
-   \param xwire,ywire x/y wire value
-
-   \return -1 on error
-   \return 0 on success
- */
-int GS_set_drawres(int id, int xres, int yres, int xwire, int ywire)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_set_drawres() id=%d xyres=%d/%d xywire=%d/%d",
-	    id, xres, yres, xwire, ywire);
-
-    if (xres < 1 || yres < 1 || xwire < 1 || ywire < 1) {
-	return (-1);
-    }
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	if (gs->x_mod != xres || gs->y_mod != yres) {
-	    gs->norm_needupdate = 1;
-	}
-
-	gs->x_mod = xres;
-	gs->y_mod = yres;
-	gs->x_modw = xwire;
-	gs->y_modw = ywire;
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Get draw resolution of surface
-
-   \param id surface id
-   \param[out] xres,yres x/y resolution value
-   \param[out] xwire,ywire x/y wire value
- */
-void GS_get_drawres(int id, int *xres, int *yres, int *xwire, int *ywire)
-{
-    geosurf *gs;
-
-    G_debug(3, "GS_get_drawres");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*xres = gs->x_mod;
-	*yres = gs->y_mod;
-	*xwire = gs->x_modw;
-	*ywire = gs->y_modw;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get dimension of surface
-
-   \param id surface id
-   \param[out] rows,cols number of rows/cols
- */
-void GS_get_dims(int id, int *rows, int *cols)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*rows = gs->rows;
-	*cols = gs->cols;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get exag-value guess
-
-   Use no_zero range because if zero IS data, then range won't be that
-   much off (it's just a GUESS, after all), but if zero is NO data, could
-   drastically affect guess
-
-   \param id surface id
-   \param[out] exag exag value
-
-   \return 1 on success
-   \return -1 on error
- */
-int GS_get_exag_guess(int id, float *exag)
-{
-    geosurf *gs;
-    float guess;
-
-    gs = gs_get_surf(id);
-    guess = 1.0;
-
-    /* if gs is type const return guess = 1.0 */
-    if (CONST_ATT == gs_get_att_src(gs, ATT_TOPO)) {
-	*exag = guess;
-	return (1);
-    }
-
-    if (gs) {
-	if (gs->zrange_nz == 0.0) {
-	    *exag = 0.0;
-
-	    return (1);
-	}
-
-	G_debug(3, "GS_get_exag_guess(): %f %f", gs->zrange_nz, Longdim);
-
-	while (gs->zrange_nz * guess / Longdim >= .25) {
-	    guess *= .1;
-
-	    G_debug(3, "GS_get_exag_guess(): %f", guess);
-	}
-
-	while (gs->zrange_nz * guess / Longdim < .025) {
-	    guess *= 10.;
-
-	    G_debug(3, "GS_get_exag_guess(): %f", guess);
-	}
-
-	*exag = guess;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get Z extents for all loaded surfaces
-
-   Treating zeros as "no data"
-
-   \param[out] min min value
-   \param[out] max max value
- */
-void GS_get_zrange_nz(float *min, float *max)
-{
-    int i, first = 1;
-    geosurf *gs;
-
-    for (i = 0; i < Next_surf; i++) {
-	if ((gs = gs_get_surf(Surf_ID[i]))) {
-	    if (first) {
-		first = 0;
-		*min = gs->zmin_nz;
-		*max = gs->zmax_nz;
-	    }
-
-	    if (gs->zmin_nz < *min) {
-		*min = gs->zmin_nz;
-	    }
-
-	    if (gs->zmax_nz > *max) {
-		*max = gs->zmax_nz;
-	    }
-	}
-    }
-
-    G_debug(3, "GS_get_zrange_nz(): min=%g max=%g", *min, *max);
-
-    return;
-}
-
-/*!
-   \brief Set translation (surface position)
-
-   \param id surface id
-   \param xtrans,ytrans,ztrans translation values
- */
-void GS_set_trans(int id, float xtrans, float ytrans, float ztrans)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	gs->x_trans = xtrans;
-	gs->y_trans = ytrans;
-	gs->z_trans = ztrans;
-    }
-
-    G_debug(3, "GS_set_trans(): id=%d, x=%f, y=%f, z=%f",
-	    id, xtrans, ytrans, ztrans);
-
-    return;
-}
-
-/*!
-   \brief Get translation values (surface position)
-
-   \param id surface id
-   \param[out] xtrans,ytrans,ztrans trans values
- */
-void GS_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
-{
-    geosurf *gs;
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	*xtrans = gs->x_trans;
-	*ytrans = gs->y_trans;
-	*ztrans = gs->z_trans;
-    }
-
-    G_debug(3, "GS_get_trans: id=%d, x=%f, y=%f, z=%f",
-	    id, *xtrans, *ytrans, *ztrans);
-
-    return;
-}
-
-
-/*!
-   \brief Get default draw color
-
-   \return color value
- */
-unsigned int GS_default_draw_color(void)
-{
-
-    G_debug(3, "GS_default_draw_color");
-
-    return ((unsigned int)Gd.bgcol);
-}
-
-/*!
-   \brief Get background color
-
-   \return color value
- */
-unsigned int GS_background_color(void)
-{
-    return ((unsigned int)Gd.bgcol);
-}
-
-/*!
-   \brief Sets which buffer to draw to
-
-   \param where GSD_BOTH, GSD_FRONT, GSD_BACK
- */
-void GS_set_draw(int where)
-{
-    Buffermode = where;
-
-    switch (where) {
-    case GSD_BOTH:
-	gsd_bothbuffers();
-
-	break;
-    case GSD_FRONT:
-	gsd_frontbuffer();
-
-	break;
-    case GSD_BACK:
-    default:
-	gsd_backbuffer();
-
-	break;
-    }
-
-    return;
-}
-
-/*
-   \brief Ready to draw
- */
-void GS_ready_draw(void)
-{
-
-    G_debug(3, "GS_ready_draw");
-
-    gsd_set_view(&Gv, &Gd);
-
-    return;
-}
-
-/*!
-   \brief Draw done, swap buffers
- */
-void GS_done_draw(void)
-{
-
-    G_debug(3, "GS_done_draw");
-
-    if (GSD_BACK == Buffermode) {
-	gsd_swapbuffers();
-    }
-
-    gsd_flush();
-
-    return;
-}
-
-/*!
-   \brief Set focus
-
-   \param realto real coordinates to
- */
-void GS_set_focus(float *realto)
-{
-
-    G_debug(3, "GS_set_focus(): %f,%f,%f", realto[0], realto[1], realto[2]);
-
-    Gv.infocus = 1;
-    GS_v3eq(Gv.real_to, realto);
-
-    gsd_set_view(&Gv, &Gd);
-
-    return;
-}
-
-/*!
-   \brief Set real focus
-
-   \param realto real coordinates to
- */
-void GS_set_focus_real(float *realto)
-{
-
-    G_get_set_window(&wind);
-    realto[X] = realto[X] - wind.west - (wind.ew_res / 2.);
-    realto[Y] = realto[Y] - wind.south - (wind.ns_res / 2.);
-
-    Gv.infocus = 1;
-    GS_v3eq(Gv.real_to, realto);
-
-    gsd_set_view(&Gv, &Gd);
-
-    return;
-}
-
-
-/*!
-   \brief Get focus
-
-   OK to call with NULL argument if just want to check state
-
-   \param realto real coordinates to
-
-   \return ?
- */
-int GS_get_focus(float *realto)
-{
-
-    G_debug(3, "GS_get_focus");
-
-    if (Gv.infocus) {
-	if (realto) {
-	    GS_v3eq(realto, Gv.real_to);
-	}
-    }
-
-    return (Gv.infocus);
-}
-
-/*!
-   \brief Set focus to map center
-
-   \param id surface id
- */
-void GS_set_focus_center_map(int id)
-{
-    float center[3];
-    geosurf *gs;
-
-    G_debug(3, "GS_set_focus_center_map");
-
-    gs = gs_get_surf(id);
-
-    if (gs) {
-	center[X] = (gs->xmax - gs->xmin) / 2.;
-	center[Y] = (gs->ymax - gs->ymin) / 2.;
-	center[Z] = (gs->zmax_nz + gs->zmin_nz) / 2.;
-
-	/* not yet working
-	   buff = gs_get_att_typbuff(gs, ATT_TOPO, 0);
-	   offset = gs->rows*gs->cols/2 + gs->cols/2;
-	   if (buff)
-	   {
-	   if (GET_MAPATT(buff, offset, tmp))
-	   {
-	   center[Z] = tmp;
-	   }
-	   }
-	 */
-
-	GS_set_focus(center);
-    }
-}
-
-/*!
-   \brief Move viewpoint 
-
-   \param pt 'from' model coordinates
- */
-void GS_moveto(float *pt)
-{
-    float ft[3];
-
-    G_debug(3, "GS_moveto(): %f,%f,%f", pt[0], pt[1], pt[2]);
-
-    if (Gv.infocus) {
-	GS_v3eq(Gv.from_to[FROM], pt);
-	/*
-	   GS_v3eq(Gv.from_to[TO], Gv.real_to);
-	 */
-	GS_v3normalize(Gv.from_to[FROM], Gv.from_to[TO]);
-	/* update inclination, look_dir if we're keeping these */
-    }
-    else {
-	GS_v3eq(ft, Gv.from_to[TO]);
-	GS_v3sub(ft, Gv.from_to[FROM]);
-	GS_v3eq(Gv.from_to[FROM], pt);
-	GS_v3eq(Gv.from_to[TO], pt);
-	GS_v3add(Gv.from_to[TO], ft);
-    }
-
-    return;
-}
-
-/*!
-   \brief Move position to (real)
-
-   \param pt point real coordinates
- */
-void GS_moveto_real(float *pt)
-{
-    gsd_real2model(pt);
-    GS_moveto(pt);
-
-    return;
-}
-
-/*!
-   \brief Get z-extent for a single surface
-
-   \param id surface id
-   \param[out] min min z-value
-   \param[out] max max z-value
-   \param[out] mid middle z-value
-
-   \return -1 on error (invalid surface id)
-   \return ?
- */
-int GS_get_zextents(int id, float *min, float *max, float *mid)
-{
-    geosurf *gs;
-
-    if (NULL == (gs = gs_get_surf(id))) {
-	return (-1);
-    }
-
-    G_debug(3, "GS_get_zextents(): id=%d", id);
-
-    return (gs_get_zextents(gs, min, max, mid));
-}
-
-/*!
-   \brief Get z-extent for all loaded surfaces
-
-   \param[out] min min z-value
-   \param[out] max max z-value
-   \param doexag use z-exaggeration
-
-   \return 1 on success
-   \return -1 on error
- */
-int GS_get_zrange(float *min, float *max, int doexag)
-{
-    int ret_surf, ret_vol;
-    float surf_min, surf_max;
-    float vol_min, vol_max;
-
-    ret_surf = gs_get_zrange(&surf_min, &surf_max);
-    ret_vol = gvl_get_zrange(&vol_min, &vol_max);
-
-    if (ret_surf > 0 && ret_vol > 0) {
-	*min = (surf_min < vol_min) ? surf_min : vol_min;
-	*max = (surf_max < vol_max) ? surf_max : vol_max;
-    }
-    else if (ret_surf > 0) {
-	*min = surf_min;
-	*max = surf_max;
-    }
-    else if (ret_vol > 0) {
-	*min = vol_min;
-	*max = vol_max;
-    }
-
-    if (doexag) {
-	*min *= Gv.vert_exag;
-	*max *= Gv.vert_exag;
-    }
-
-    G_debug(3, "GS_get_zrange(): min=%g max=%g", *min, *max);
-    return ((ret_surf > 0 || ret_vol > 0) ? (1) : (-1));
-}
-
-/*!
-   \brief Get viewpoint 'from' position
-
-   \param[out] fr from model coordinates
- */
-void GS_get_from(float *fr)
-{
-    GS_v3eq(fr, Gv.from_to[FROM]);
-
-    G_debug(3, "GS_get_from(): %f,%f,%f", fr[0], fr[1], fr[2]);
-
-    return;
-}
-
-/*!
-   \brief Get viewpoint 'from' real coordinates
-
-   \param[out] fr 'from' real coordinates
- */
-void GS_get_from_real(float *fr)
-{
-    GS_v3eq(fr, Gv.from_to[FROM]);
-    gsd_model2real(fr);
-
-    return;
-}
-
-/*!
-   \brief Get 'to' real coordinates
-
-   \param[out] to 'to' real coordinates
- */
-void GS_get_to_real(float *to)
-{
-    float realto[3];
-
-    G_get_set_window(&wind);
-    GS_get_focus(realto);
-    to[X] = realto[X] + wind.west + (wind.ew_res / 2.);
-    to[Y] = realto[Y] + wind.south + (wind.ns_res / 2.);
-    to[Z] = realto[Z];
-
-    return;
-}
-
-
-/*!
-   \brief Get zoom setup
-
-   \param[out] a,b,c,d current viewport settings
-   \param[out] maxx,maxy max viewport size
- */
-void GS_zoom_setup(int *a, int *b, int *c, int *d, int *maxx, int *maxy)
-{
-    GLint tmp[4];
-    GLint num[2];
-
-    gsd_getViewport(tmp, num);
-    *a = tmp[0];
-    *b = tmp[1];
-    *c = tmp[2];
-    *d = tmp[3];
-    *maxx = num[0];
-    *maxy = num[1];
-
-    return;
-}
-
-/*!
-   \brief Get 'to' model coordinates
-
-   \todo need set_to? - just use viewdir?
-
-   \param[out] to 'to' model coordinates
- */
-void GS_get_to(float *to)
-{
-    G_debug(3, "GS_get_to");
-
-    GS_v3eq(to, Gv.from_to[TO]);
-
-    return;
-}
-
-/*!
-   \brief Get viewdir
-
-   \param[out] dir viewdir value
- */
-void GS_get_viewdir(float *dir)
-{
-    GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], dir);
-
-    return;
-}
-
-/*!
-   \brief Set viewdir
-
-   Automatically turns off focus
-
-   \param dir viewdir value
- */
-void GS_set_viewdir(float *dir)
-{
-    float tmp[3];
-
-    GS_v3eq(tmp, dir);
-    GS_v3norm(tmp);
-    GS_v3eq(Gv.from_to[TO], Gv.from_to[FROM]);
-    GS_v3add(Gv.from_to[TO], tmp);
-
-    GS_set_nofocus();
-    gsd_set_view(&Gv, &Gd);
-
-    return;
-}
-
-/*!
-   \brief Set field of view
-
-   \param fov fov value
- */
-void GS_set_fov(int fov)
-{
-    Gv.fov = fov;
-
-    return;
-}
-
-/*!
-   \brief Get fied of view
-
-   \return field of view, in 10ths of degrees
- */
-int GS_get_fov(void)
-{
-    return (Gv.fov);
-}
-
-/*!
-   \brief Get twist value
-
-   10ths of degrees off twelve o'clock
- */
-int GS_get_twist(void)
-{
-    return (Gv.twist);
-}
-
-/*!
-   \brief Set viewpoint twist value
-
-   10ths of degrees off twelve o'clock
-
-   \param t tenths of degrees clockwise from 12:00.
- */
-void GS_set_twist(int t)
-{
-    Gv.twist = t;
-
-    return;
-}
-
-/*!
-   \brief Set rotation params
- */
-void GS_set_rotation(double angle, double x, double y, double z)
-{
-    Gv.rotate.rot_angle = angle;
-    Gv.rotate.rot_axes[0] = x;
-    Gv.rotate.rot_axes[1] = y;
-    Gv.rotate.rot_axes[2] = z;
-    Gv.rotate.do_rot = 1;
-
-    return;
-}
-
-/*!
-   \brief Stop scene rotation
- */
-void GS_unset_rotation(void)
-{
-    Gv.rotate.do_rot = 0;
-}
-
-/*!
-   \brief Reset scene rotation
- */
-void GS_init_rotation(void)
-{
-    int i;
-
-    for (i = 0; i < 16; i++) {
-	if (i == 0 || i == 5 || i == 10 || i == 15)
-	    Gv.rotate.rotMatrix[i] = 1.0;
-	else
-	    Gv.rotate.rotMatrix[i] = 0.0;
-    }
-    Gv.rotate.rot_angle = 0.0;
-    Gv.rotate.rot_axes[0] = 0.0;
-    Gv.rotate.rot_axes[1] = 0.0;
-    Gv.rotate.rot_axes[2] = 0.0;
-    Gv.rotate.do_rot = 0;
-    
-}
-/*!
- * \brief Get rotation matrix
- */ 
-void GS_get_rotation_matrix(double *matrix)
-{
-    int i;
-
-    for (i = 0; i < 16; i++) {
-	matrix[i] = Gv.rotate.rotMatrix[i];
-    }
-}
-
-/*!
- * \brief Set rotation matrix
- */ 
-void GS_set_rotation_matrix(double *matrix)
-{
-    int i;
-
-    for (i = 0; i < 16; i++) {
-	Gv.rotate.rotMatrix[i] = matrix[i];
-    }
-}
-
-/*!
-   \brief Unset focus
- */
-void GS_set_nofocus(void)
-{
-    G_debug(3, "GS_set_nofocus");
-
-    Gv.infocus = 0;
-
-    return;
-}
-
-/*!
-   \brief Set focus
-
-   Make sure that the center of view is set
- */
-void GS_set_infocus(void)
-{
-    G_debug(3, "GS_set_infocus");
-
-    Gv.infocus = 1;
-
-    return;
-}
-
-/*!
-   \brief Set viewport
-
-   \param left,right,bottom,top viewport extent values
- */
-void GS_set_viewport(int left, int right, int bottom, int top)
-{
-    G_debug(3, "GS_set_viewport(): left=%d, right=%d, "
-	    "bottom=%d, top=%d", left, right, bottom, top);
-
-    gsd_viewport(left, right, bottom, top);
-
-    return;
-}
-
-/*!
-   \brief Send screen coords sx and sy, lib traces through surfaces; sets
-   new center to point of nearest intersection.
-
-   If no intersection, uses line of sight with length of current view
-   ray (eye to center) to set new center.
-
-   Reset center of view to screen coordinates sx, sy.
-
-   \param sx,sy screen coordinates
-
-   \return 1 on success
-   \return 0 on error (invalid surface id)
- */
-int GS_look_here(int sx, int sy)
-{
-    float x, y, z, len, los[2][3];
-    Point3 realto, dir;
-    int id;
-    geosurf *gs;
-
-    if (GS_get_selected_point_on_surface(sx, sy, &id, &x, &y, &z)) {
-	gs = gs_get_surf(id);
-	if (gs) {
-	    realto[X] = x - gs->ox + gs->x_trans;
-	    realto[Y] = y - gs->oy + gs->y_trans;
-	    realto[Z] = z + gs->z_trans;
-	    GS_set_focus(realto);
-
-	    return (1);
-	}
-    }
-    else {
-	if (gsd_get_los(los, (short)sx, (short)sy)) {
-	    len = GS_distance(Gv.from_to[FROM], Gv.real_to);
-	    GS_v3dir(los[FROM], los[TO], dir);
-	    GS_v3mult(dir, len);
-	    realto[X] = Gv.from_to[FROM][X] + dir[X];
-	    realto[Y] = Gv.from_to[FROM][Y] + dir[Y];
-	    realto[Z] = Gv.from_to[FROM][Z] + dir[Z];
-	    GS_set_focus(realto);
-
-	    return (1);
-	}
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Get selected point of surface
-
-   Given screen coordinates sx and sy, find closest intersection of
-   view ray with surfaces and return coordinates of intersection in x, y,
-   z, and identifier of surface in id.
-
-   \param sx,sy screen coordinates
-   \param[out] id surface id
-   \param[out] x,y,z point on surface (model coordinates?)
-
-   \returns 0 if no intersections found
-   \return number of intersections
- */
-int GS_get_selected_point_on_surface(int sx, int sy, int *id, float *x,
-				     float *y, float *z)
-{
-    float los[2][3], find_dist[MAX_SURFS], closest;
-    Point3 point, tmp, finds[MAX_SURFS];
-    int surfs[MAX_SURFS], i, iclose, numhits = 0;
-    geosurf *gs;
-
-    /* returns surface-world coords */
-    gsd_get_los(los, (short)sx, (short)sy);
-
-    if (!gs_setlos_enterdata(los)) {
-	G_debug(3, "gs_setlos_enterdata(los): returns false");
-	return (0);
-    }
-
-    for (i = 0; i < Next_surf; i++) {
-	G_debug(3, "id=%d", i);
-
-	gs = gs_get_surf(Surf_ID[i]);
-
-	/* los_intersect expects surf-world coords (xy transl, no scaling) */
-
-#if NVIZ_HACK
-	if (gs_los_intersect1(Surf_ID[i], los, point)) {
-#else
-	if (gs_los_intersect(Surf_ID[i], los, point)) {
-#endif
-	    if (!gs_point_is_masked(gs, point)) {
-		GS_v3eq(tmp, point);
-		tmp[X] += gs->x_trans;
-		tmp[Y] += gs->y_trans;
-		tmp[Z] += gs->z_trans;
-		find_dist[numhits] = GS_distance(los[FROM], tmp);
-		gsd_surf2real(gs, point);
-		GS_v3eq(finds[numhits], point);
-		surfs[numhits] = Surf_ID[i];
-		numhits++;
-	    }
-	}
-    }
-
-    for (i = iclose = 0; i < numhits; i++) {
-	closest = find_dist[iclose];
-
-	if (find_dist[i] < closest) {
-	    iclose = i;
-	}
-    }
-
-    if (numhits) {
-	*x = finds[iclose][X];
-	*y = finds[iclose][Y];
-	*z = finds[iclose][Z];
-	*id = surfs[iclose];
-    }
-
-    G_debug(3, "NumHits %d, next %d", numhits, Next_surf);
-
-    return (numhits);
-}
-
-/*!
-   \brief Set cplace rotation
-
-   \param num cplace id
-   \param dx,dy,dz rotation values
- */
-void GS_set_cplane_rot(int num, float dx, float dy, float dz)
-{
-    gsd_cplane_setrot(num, dx, dy, dz);
-
-    return;
-}
-
-/*!
-   \brief Set cplace trans
-
-   \param num cplace id
-   \param dx,dy,dz rotation values
- */
-void GS_set_cplane_trans(int num, float dx, float dy, float dz)
-{
-    gsd_cplane_settrans(num, dx, dy, dz);
-
-    return;
-}
-
-
-/*!
-   \brief Draw cplace
-
-   \param num cplace id
- */
-void GS_draw_cplane(int num)
-{
-    geosurf *gsurfs[MAX_SURFS];
-    int nsurfs;
-
-    nsurfs = gs_num_surfaces();
-    if (2 == nsurfs) {
-	/* testing */
-	gs_getall_surfaces(gsurfs);
-	gsd_draw_cplane_fence(gsurfs[0], gsurfs[1], num);
-    }
-    else {
-	gsd_draw_cplane(num);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw cplace fence ?
-
-   \param hs1,hs2
-   \param num cplane id
-
-   \return 0 on error
-   \return 1 on success
- */
-int GS_draw_cplane_fence(int hs1, int hs2, int num)
-{
-    geosurf *gs1, *gs2;
-
-    if (NULL == (gs1 = gs_get_surf(hs1))) {
-	return (0);
-    }
-
-    if (NULL == (gs2 = gs_get_surf(hs2))) {
-	return (0);
-    }
-
-    gsd_draw_cplane_fence(gs1, gs2, num);
-
-    return (1);
-}
-
-/*!
-   \brief Draw all cplace fences ?
- */
-void GS_alldraw_cplane_fences(void)
-{
-    int onstate[MAX_CPLANES], i;
-
-    gsd_get_cplanes_state(onstate);
-
-    for (i = 0; i < MAX_CPLANES; i++) {
-	if (onstate[i]) {
-	    GS_draw_cplane_fence(Surf_ID[0], Surf_ID[1], i);
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Set cplace
-
-   \param num cplane id
- */
-void GS_set_cplane(int num)
-{
-    gsd_cplane_on(num);
-
-    return;
-}
-
-/*!
-   \brief Unset clip place (turn off)
-
-   \param num cplane id
- */
-void GS_unset_cplane(int num)
-{
-    gsd_cplane_off(num);
-
-    return;
-}
-
-/*!
-   \brief Get axis scale
-
-   \param sx,sy,sz x/y/z scale values
-   \param doexag use vertical exaggeration
- */
-void GS_get_scale(float *sx, float *sy, float *sz, int doexag)
-{
-    float zexag;
-
-    zexag = doexag ? Gv.vert_exag : 1.;
-    *sx = *sy = Gv.scale;
-    *sz = Gv.scale * zexag;
-
-    return;
-}
-
-/*!
-   \brief Set fence color
-
-   \param mode mode id
- */
-void GS_set_fencecolor(int mode)
-{
-    gsd_setfc(mode);
-
-    return;
-}
-
-/*!
-   \brief Get fence color
-
-   \return color value
- */
-int GS_get_fencecolor(void)
-{
-    return gsd_getfc();
-}
-
-/*!
-   \brief Measure distance "as the ball rolls" between two points on
-   surface
-
-   \param hs surface id
-   \param x1,y1,x2,y2 two points on surface
-   \param[out] dist measured distance
-   \param use_exag use exag. surface
-
-   \return 0 on error or if one or more points is not in region
-   \return distance following terrain
- */
-int GS_get_distance_alongsurf(int hs, float x1, float y1, float x2, float y2,
-			      float *dist, int use_exag)
-{
-    geosurf *gs;
-    float p1[2], p2[2];
-    
-    gs = gs_get_surf(hs);
-    if (gs == NULL) {
-	return 0;
-    }
-    
-    p1[X] = x1;
-    p1[Y] = y1;
-    p2[X] = x2;
-    p2[Y] = y2;
-    gsd_real2surf(gs, p1);
-    gsd_real2surf(gs, p2);
-
-    G_debug(3, "GS_get_distance_alongsurf(): hs=%d p1=%f,%f p2=%f,%f",
-	    hs, x1, y1, x2, y2);
-    return gs_distance_onsurf(gs, p1, p2, dist, use_exag);
-}
-
-/*!
-   \brief Save 3d view
-
-   \param vname view file name
-   \param surfid surface id
-
-   \return ?
- */
-int GS_save_3dview(const char *vname, int surfid)
-{
-    return (Gs_save_3dview(vname, &Gv, &Gd, &wind, gs_get_surf(surfid)));
-}
-
-/*!
-   \brief Load 3d view
-
-   \param vname view file name
-   \param surfid surface id
-
-   \return ?
- */
-int GS_load_3dview(const char *vname, int surfid)
-{
-
-    return (Gs_load_3dview(vname, &Gv, &Gd, &wind, gs_get_surf(surfid)));
-
-    /* what to do about lights - I guess, delete all &
-       create any that exist in 3dview file */
-}
-
-/************************************************************************
-* Following routines use Graphics Library
-************************************************************************/
-
-/*!
-   \brief Init viewpoint
-
-   \todo allow to set center?
- */
-void GS_init_view(void)
-{
-    static int first = 1;
-
-    G_debug(3, "GS_init_view");
-
-    if (first) {
-	first = 0;
-	glMatrixMode(GL_MODELVIEW);
-
-	/* OGLXXX doublebuffer: use GLX_DOUBLEBUFFER in attriblist */
-	/* glxChooseVisual(*dpy, screen, *attriblist); */
-	/* OGLXXX
-	 * ZMIN not needed -- always 0.
-	 * ZMAX not needed -- always 1.
-	 * getgdesc other posiblilties:
-	 *      glxGetConfig();
-	 *      glxGetCurrentContext();
-	 *      glxGetCurrentDrawable();
-	 * GLint gdtmp;
-	 * getgdesc other posiblilties:
-	 *      glxGetConfig();
-	 *      glxGetCurrentContext();
-	 *      glxGetCurrentDrawable();
-	 * GLint gdtmp;
-	 * glDepthRange params must be scaled to [0, 1]
-	 */
-	glDepthRange(0.0, 1.0);
-	glEnable(GL_DEPTH_TEST);
-	glDepthFunc(GL_LEQUAL);
-	/* } */
-
-	/* replace these with something meaningful */
-	Gv.fov = 450;
-	Gv.twist = 0;
-
-	GS_init_rotation();
-
-	Gv.from_to[FROM][X] = Gv.from_to[FROM][Y] =
-	    Gv.from_to[FROM][Z] = GS_UNIT_SIZE / 2.;
-
-	Gv.from_to[TO][X] = GS_UNIT_SIZE / 2.;
-	Gv.from_to[TO][Y] = GS_UNIT_SIZE / 2.;
-	Gv.from_to[TO][Z] = 0.;
-	Gv.from_to[TO][W] = Gv.from_to[FROM][W] = 1.;
-
-	Gv.real_to[W] = 1.;
-	Gv.vert_exag = 1.;
-
-	GS_v3eq(Gv.real_to, Gv.from_to[TO]);
-	GS_v3normalize(Gv.from_to[FROM], Gv.from_to[TO]);
-
-	/*
-	   Gd.nearclip = 50;
-	   Gd.farclip = 10000.;
-	 */
-	Gd.nearclip = 10.;
-	Gd.farclip = 10000.;
-	Gd.aspect = (float)GS_get_aspect();
-
-	GS_set_focus(Gv.real_to);
-    }
-
-    return;
-}
-
-/*!
-   \brief Clear view
-
-   \param col color value
- */
-void GS_clear(int col)
-{
-    G_debug(3, "GS_clear");
-
-    col = col | 0xFF000000;
-
-    /* OGLXXX
-     * change glClearDepth parameter to be in [0, 1]
-     * ZMAX not needed -- always 1.
-     * getgdesc other posiblilties:
-     *      glxGetConfig();
-     *      glxGetCurrentContext();
-     *      glxGetCurrentDrawable();
-     * GLint gdtmp;
-     */
-    glClearDepth(1.0);
-    glClearColor(((float)((col) & 0xff)) / 255.,
-		 (float)((col) >> 8 & 0xff) / 255.,
-		 (float)((col) >> 16 & 0xff) / 255.,
-		 (float)((col) >> 24 & 0xff) / 255.);
-    glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT);
-
-    Gd.bgcol = col;
-    Modelshowing = 0;
-    gsd_flush();
-
-    return;
-}
-
-/*!
-   \brief Get aspect value
-
-   \return aspect value
- */
-double GS_get_aspect(void)
-{
-    int left, right, bottom, top;
-    GLint tmp[4];
-
-    /* OGLXXX
-     * get GL_VIEWPORT:
-     * You can probably do better than this.
-     */
-    glGetIntegerv(GL_VIEWPORT, tmp);
-    left = tmp[0];
-    right = tmp[0] + tmp[2] - 1;
-    bottom = tmp[1];
-    top = tmp[1] + tmp[3] - 1;
-
-    G_debug(3, "GS_get_aspect(): left=%d, right=%d, top=%d, bottom=%d",
-	    left, right, top, bottom);
-
-    return ((double)(right - left) / (top - bottom));
-}
-
-/*!
-   \brief Check for transparency
-
-   Disabled.
-
-   \return 1
- */
-int GS_has_transparency(void)
-{
-    /* OGLXXX
-     * getgdesc other posiblilties:
-     *      glxGetConfig();
-     *      glxGetCurrentContext();
-     *      glxGetCurrentDrawable();
-     * GLint gdtmp;
-     * blending is ALWAYS supported.
-     * This function returns whether it is enabled.
-     * return((glGetIntegerv(GL_BLEND, &gdtmp), gdtmp));
-     */
-
-    return (1);
-}

Deleted: grass/trunk/lib/ogsf/GSX.c
===================================================================
--- grass/trunk/lib/ogsf/GSX.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GSX.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,70 +0,0 @@
-/*!
-   \file GSX.c
-
-   \brief OGSF library - loading and manipulating surfaces
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown USACERL (December 1993)
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <grass/ogsf.h>
-
-void (*Cxl_func) ();
-void (*Swap_func) ();
-
-static int Cxl = 0;
-
-/*!
-   \brief Check for cancel
-
-   \return code
- */
-int GS_check_cancel(void)
-{
-    Cxl_func();
-
-    return (Cxl);
-}
-
-/*!
-   \brief Set cancel
- */
-void GS_set_cancel(int c)
-{
-    Cxl = c;
-
-    return;
-}
-
-/*!
-   \brief Set cxl function
-
-   \param pointer to function
- */
-void GS_set_cxl_func(void (*f) (void))
-{
-    Cxl_func = f;
-
-    return;
-}
-
-/*!
-   \brief Set swap function
-
-   \param pointer to function
- */
-void GS_set_swap_func(void (*f) (void))
-{
-    Swap_func = f;
-
-    return;
-}

Deleted: grass/trunk/lib/ogsf/GS_util.c
===================================================================
--- grass/trunk/lib/ogsf/GS_util.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GS_util.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,486 +0,0 @@
-/*!
-   \file GS_util.c
-
-   \brief OGSF library - loading and manipulating surfaces
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown USACERL, GMSL/University of Illinois
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <stdlib.h>
-#include <math.h>
-#include <string.h>
-
-#include <grass/gis.h>
-#include <grass/ogsf.h>
-
-/*!
-   \brief Calculate distance between 2 coordinates
-
-   Units is one of:
-   - "meters",
-   - "miles",
-   - "kilometers",
-   - "feet",
-   - "yards",
-   - "nmiles" (nautical miles),
-   - "rods",
-   - "inches",
-   - "centimeters",
-   - "millimeters",
-   - "micron",
-   - "nanometers",
-   - "cubits",
-   - "hands",
-   - "furlongs",
-   - "chains"
-
-   Default is meters.
-
-   \param from starting point
-   \param to ending point
-   \param units map units
-
-   \return distance between two geographic coordinates in current projection
- */
-double GS_geodistance(double *from, double *to, const char *units)
-{
-    double meters;
-
-    meters = Gs_distance(from, to);
-
-    if (!units) {
-	return (meters);
-    }
-
-    if (strcmp(units, "meters") == 0) {
-	return (meters);
-    }
-
-    if (strcmp(units, "miles") == 0) {
-	return (meters * .0006213712);
-    }
-
-    if (strcmp(units, "kilometers") == 0) {
-	return (meters * .001);
-    }
-
-    if (strcmp(units, "feet") == 0) {
-	return (meters * 3.280840);
-    }
-
-    if (strcmp(units, "yards") == 0) {
-	return (meters * 1.093613);
-    }
-
-    if (strcmp(units, "rods") == 0) {
-	return (meters * .1988388);
-    }
-
-    if (strcmp(units, "inches") == 0) {
-	return (meters * 39.37008);
-    }
-
-    if (strcmp(units, "centimeters") == 0) {
-	return (meters * 100.0);
-    }
-
-    if (strcmp(units, "millimeters") == 0) {
-	return (meters * 1000.0);
-    }
-
-    if (strcmp(units, "micron") == 0) {
-	return (meters * 1000000.0);
-    }
-
-    if (strcmp(units, "nanometers") == 0) {
-	return (meters * 1000000000.0);
-    }
-
-    if (strcmp(units, "cubits") == 0) {
-	return (meters * 2.187227);
-    }
-
-    if (strcmp(units, "hands") == 0) {
-	return (meters * 9.842520);
-    }
-
-    if (strcmp(units, "furlongs") == 0) {
-	return (meters * .004970970);
-    }
-
-    if (strcmp(units, "nmiles") == 0) {
-	/* nautical miles */
-	return (meters * .0005399568);
-    }
-
-    if (strcmp(units, "chains") == 0) {
-	return (meters * .0497097);
-    }
-
-    return (meters);
-}
-
-/*!
-   \brief Calculate distance
-
-   \param from 'from' point (X,Y,Z)
-   \param to 'to' point (X,Y,Z)
-
-   \return distance
- */
-float GS_distance(float *from, float *to)
-{
-    float x, y, z;
-
-    x = from[X] - to[X];
-    y = from[Y] - to[Y];
-    z = from[Z] - to[Z];
-
-    return (float)sqrt(x * x + y * y + z * z);
-}
-
-/*!
-   \brief Calculate distance in plane
-
-   \param from 'from' point (X,Y)
-   \param to 'to' point (X,Y)
-
-   \return distance
- */
-float GS_P2distance(float *from, float *to)
-{
-    float x, y;
-
-    x = from[X] - to[X];
-    y = from[Y] - to[Y];
-
-    return (float)sqrt(x * x + y * y);
-}
-
-/*!
-   \brief Copy vector values
-
-   v1 = v2
-
-   \param[out] v1 first vector
-   \param v2 second vector
- */
-void GS_v3eq(float *v1, float *v2)
-{
-    v1[X] = v2[X];
-    v1[Y] = v2[Y];
-    v1[Z] = v2[Z];
-
-    return;
-}
-
-/*!
-   \brief Sum vectors
-
-   v1 += v2
-
-   \param[in,out] v1 first vector
-   \param v2 second vector
- */
-void GS_v3add(float *v1, float *v2)
-{
-    v1[X] += v2[X];
-    v1[Y] += v2[Y];
-    v1[Z] += v2[Z];
-
-    return;
-}
-
-/*!
-   \brief Subtract vectors
-
-   v1 -= v2
-
-   \param[in,out] v1 first vector
-   \param v2 second vector
- */
-void GS_v3sub(float *v1, float *v2)
-{
-    v1[X] -= v2[X];
-    v1[Y] -= v2[Y];
-    v1[Z] -= v2[Z];
-
-    return;
-}
-
-/*!
-   \brief Multiple vectors
-
-   v1 *= k
-
-   \param[in,out] v1 vector
-   \param k multiplicator
- */
-void GS_v3mult(float *v1, float k)
-{
-    v1[X] *= k;
-    v1[Y] *= k;
-    v1[Z] *= k;
-
-    return;
-}
-
-/*!
-   \brief Change v1 so that it is a unit vector (2D)
-
-   \param[in,out] v1 vector
-
-   \return 0 if magnitude of v1 is zero
-   \return 1 if magnitude of v1 > 0
- */
-int GS_v3norm(float *v1)
-{
-    float n;
-
-    n = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y] + v1[Z] * v1[Z]);
-
-    if (n == 0.0) {
-	return (0);
-    }
-
-    v1[X] /= n;
-    v1[Y] /= n;
-    v1[Z] /= n;
-
-    return (1);
-}
-
-/*!
-   \brief Change v1 so that it is a unit vector (3D)
-
-   \param[in,out] v1 vector
-
-   \return 0 if magnitude of v1 is zero
-   \return 1 if magnitude of v1 > 0
- */
-int GS_v2norm(float *v1)
-{
-    float n;
-
-    n = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y]);
-
-    if (n == 0.0) {
-	return (0);
-    }
-
-    v1[X] /= n;
-    v1[Y] /= n;
-
-    return (1);
-}
-
-/*!
-   \brief Changes v1 so that it is a unit vector
-
-   \param dv1 vector
-
-   \return 0 if magnitude of dv1 is zero
-   \return 1 if magnitude of dv1 > 0
- */
-int GS_dv3norm(double *dv1)
-{
-    double n;
-
-    n = sqrt(dv1[X] * dv1[X] + dv1[Y] * dv1[Y] + dv1[Z] * dv1[Z]);
-
-    if (n == 0.0) {
-	return (0);
-    }
-
-    dv1[X] /= n;
-    dv1[Y] /= n;
-    dv1[Z] /= n;
-
-    return (1);
-}
-
-
-/*!
-   \brief Change v2 so that v1v2 is a unit vector
-
-   \param v1 first vector
-   \param v2[in,out] second vector
-
-   \return 0 if magnitude of dx is zero
-   \return 1 if magnitude of dx > 0
- */
-int GS_v3normalize(float *v1, float *v2)
-{
-    float n, dx, dy, dz;
-
-    dx = v2[X] - v1[X];
-    dy = v2[Y] - v1[Y];
-    dz = v2[Z] - v1[Z];
-    n = sqrt(dx * dx + dy * dy + dz * dz);
-
-    if (n == 0.0) {
-	return (0);
-    }
-
-    v2[X] = v1[X] + dx / n;
-    v2[Y] = v1[Y] + dy / n;
-    v2[Z] = v1[Z] + dz / n;
-
-    return (1);
-}
-
-
-/*!
-   \brief Get a normalized direction from v1 to v2, store in v3
-
-   \param v1 first vector
-   \param v2 second vector
-   \param[out] v3 output vector
-
-   \return 0 if magnitude of dx is zero
-   \return 1 if magnitude of dx > 0
- */
-int GS_v3dir(float *v1, float *v2, float *v3)
-{
-    float n, dx, dy, dz;
-
-    dx = v2[X] - v1[X];
-    dy = v2[Y] - v1[Y];
-    dz = v2[Z] - v1[Z];
-    n = sqrt(dx * dx + dy * dy + dz * dz);
-
-    if (n == 0.0) {
-	v3[X] = v3[Y] = v3[Z] = 0.0;
-	return (0);
-    }
-
-    v3[X] = dx / n;
-    v3[Y] = dy / n;
-    v3[Z] = dz / n;
-
-    return (1);
-}
-
-
-/*!
-   \brief Get a normalized direction from v1 to v2, store in v3 (2D)
-
-   \param v1 first vector
-   \param v2 second vector
-   \param[out] v3 output vector
-
-   \return 0 if magnitude of dx is zero
-   \return 1 if magnitude of dx > 0
- */
-void GS_v2dir(float *v1, float *v2, float *v3)
-{
-    float n, dx, dy;
-
-    dx = v2[X] - v1[X];
-    dy = v2[Y] - v1[Y];
-    n = sqrt(dx * dx + dy * dy);
-
-    v3[X] = dx / n;
-    v3[Y] = dy / n;
-
-    return;
-}
-
-/*!
-   \brief Get the cross product v3 = v1 cross v2
-
-   \param v1 first vector
-   \param v2 second vector
-   \param[out] v3 output vector
- */
-void GS_v3cross(float *v1, float *v2, float *v3)
-{
-    v3[X] = (v1[Y] * v2[Z]) - (v1[Z] * v2[Y]);
-    v3[Y] = (v1[Z] * v2[X]) - (v1[X] * v2[Z]);
-    v3[Z] = (v1[X] * v2[Y]) - (v1[Y] * v2[X]);
-
-    return;
-}
-
-/*!
-   \brief Magnitude of vector
-
-   \param v1 vector
-   \param[out] mag magnitude value
- */
-void GS_v3mag(float *v1, float *mag)
-{
-    *mag = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y] + v1[Z] * v1[Z]);
-
-    return;
-}
-
-/*!
-   \brief ADD
-
-   Initialize by calling with a number nhist to represent number of
-   previous entrys to check, then call with zero as nhist
-
-   \param p1 first point
-   \param p2 second point
-   \param nhist ?
-
-   \return -1 on error
-   \return -2
-   \return 1
-   \return 9
- */
-int GS_coordpair_repeats(float *p1, float *p2, int nhist)
-{
-    static float *entrys = NULL;
-    static int next = 0;
-    static int len = 0;
-    int i;
-
-    if (nhist) {
-	if (entrys) {
-	    G_free(entrys);
-	}
-
-	entrys = (float *)G_malloc(4 * nhist * sizeof(float));
-
-	if (!entrys)
-	    return (-1);
-
-	len = nhist;
-	next = 0;
-    }
-
-    if (!len) {
-	return (-2);
-    }
-
-    for (i = 0; i < next; i += 4) {
-	if (entrys[i] == p1[0] && entrys[i + 1] == p1[1]
-	    && entrys[i + 2] == p2[0] && entrys[i + 3] == p2[1]) {
-	    return (1);
-	}
-    }
-
-    if (len == next / 4) {
-	next = 0;
-    }
-
-    entrys[next] = p1[0];
-    entrys[next + 1] = p1[1];
-    entrys[next + 2] = p2[0];
-    entrys[next + 3] = p2[1];
-    next += 4;
-
-    return (0);
-}

Deleted: grass/trunk/lib/ogsf/GV2.c
===================================================================
--- grass/trunk/lib/ogsf/GV2.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GV2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,601 +0,0 @@
-/*!
-   \file lib/ogsf/GV2.c
-
-   \brief OGSF library - loading and manipulating vector sets (higher level functions)
-
-   (C) 1999-2008, 2011 by the GRASS Development Team
-
-   This program is free software under the GNU General Public License
-   (>=v2).  Read the file COPYING that comes with GRASS for details.
-
-   \author Bill Brown USACERL, GMSL/University of Illinois
-   \author Updated by Martin landa <landa.martin gmail.com>
-   (doxygenized in May 2008, thematic mapping in June 2011)
- */
-
-#include <stdlib.h>
-#include <string.h>
-
-#include <grass/gis.h>
-#include <grass/ogsf.h>
-
-#include "gsget.h"
-
-static int Vect_ID[MAX_VECTS];
-static int Next_vect = 0;
-
-/*!
-   \brief Check if vector set exists
-
-   \param id vector set id
-
-   \return 0 not found
-   \return 1 found
- */
-int GV_vect_exists(int id)
-{
-    int i, found = 0;
-
-    G_debug(3, "GV_vect_exists");
-
-    if (NULL == gv_get_vect(id)) {
-	return (0);
-    }
-
-    for (i = 0; i < Next_vect && !found; i++) {
-	if (Vect_ID[i] == id) {
-	    found = 1;
-	}
-    }
-
-    return (found);
-}
-
-/*!
-   \brief Register new vector set
-
-   \return vector set id
-   \return -1 on error
- */
-int GV_new_vector(void)
-{
-    geovect *nv;
-
-    if (Next_vect < MAX_VECTS) {
-	nv = gv_get_new_vect();
-	gv_set_defaults(nv);
-	Vect_ID[Next_vect] = nv->gvect_id;
-	++Next_vect;
-
-	G_debug(3, "GV_new_vector(): id=%d", nv->gvect_id);
-
-	return nv->gvect_id;
-    }
-
-    return -1;
-}
-
-/*!
-   \brief Get number of available vector sets
-
-   \return number of vector sets
- */
-int GV_num_vects(void)
-{
-    return (gv_num_vects());
-}
-
-/*!
-   \brief Get list of vector sets
-
-   Must free when no longer needed!
-
-   \param numvects number of vector sets
-
-   \return pointer to list of point sets
-   \return NULL on error
- */
-int *GV_get_vect_list(int *numvects)
-{
-    int i, *ret;
-
-    *numvects = Next_vect;
-
-    if (Next_vect) {
-	ret = (int *)G_malloc(Next_vect * sizeof(int));
-	if (!ret) {
-	    return (NULL);
-	}
-
-	for (i = 0; i < Next_vect; i++) {
-	    ret[i] = Vect_ID[i];
-	}
-
-	return (ret);
-    }
-
-    return (NULL);
-}
-
-/*!
-   \brief Delete vector set from list
-
-   \param id vector set id
-
-   \return 1 on success
-   \return -1 on error
- */
-int GV_delete_vector(int id)
-{
-    int i, j, found = 0;
-
-    G_debug(3, "GV_delete_vect");
-
-    if (GV_vect_exists(id)) {
-	gv_delete_vect(id);
-
-	for (i = 0; i < Next_vect && !found; i++) {
-	    if (Vect_ID[i] == id) {
-		found = 1;
-
-		for (j = i; j < Next_vect; j++) {
-		    Vect_ID[j] = Vect_ID[j + 1];
-		}
-	    }
-	}
-
-	if (found) {
-	    --Next_vect;
-	    return (1);
-	}
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Load vector set
-
-   Check to see if handle already loaded, if so - free before loading
-   new for now, always load to memory
-
-   \todo Load file handle & ready for reading instead of using
-   memory
-
-   \param id vector set id
-   \param filename filename
-
-   \return -1 on error (invalid vector set id)
-   \return 1 on success
- */
-int GV_load_vector(int id, const char *filename)
-{
-    geovect *gv;
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return (-1);
-    }
-
-    if (gv->lines) {
-	gv_free_vectmem(gv);
-    }
-
-    gv->filename = G_store(filename);
-
-    if ((gv->lines = Gv_load_vect(filename, &(gv->n_lines)))) {
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get vector map name
-
-   Note: char array is allocated by G_store()
-
-   \param id vector set id
-   \param filename &filename
-
-   \return -1 on error (invalid vector set id)
-   \return 1 on success
- */
-int GV_get_vectname(int id, char **filename)
-{
-    geovect *gv;
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return (-1);
-    }
-
-    *filename = G_store(gv->filename);
-
-    return (1);
-}
-
-/*!
-   \brief Set vector style
-
-   \param id vector set id
-   \param mem non-zero for use memory
-   \param color color value
-   \param width line width
-   \param flat non-zero for flat mode
-
-   \return -1 on error (invalid vector set id)
-   \return 1 on success
- */
-int GV_set_style(int id, int mem, int color, int width, int flat)
-{
-    geovect *gv;
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return -1;
-    }
-
-    gv->use_mem = mem;
-    gv->flat_val = flat;
-    gv->style->color = color;
-    gv->style->width = width;
-
-    return 1;
-}
-
-
-/*!
-   \brief Get vector style
-
-   \param id vector set id
-   \param[out] mem non-zero for use memory
-   \param[out] color color value
-   \param[out] width line width
-   \param[out] flat non-zero for flat mode
-
-   \return -1 on error (invalid vector set id)
-   \return 1 on success
- */
-int GV_get_style(int id, int *mem, int *color, int *width, int *flat)
-{
-    geovect *gv;
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return -1;
-    }
-
-    *mem = gv->use_mem;
-    *color = gv->style->color;
-    *width = gv->style->width;
-    *flat = gv->flat_val;
-
-    return 1;
-}
-
-/*!
-   \brief Set vector set style for thematic mapping
-   
-   Updates also style for each geoline.
-   
-   \param id vector set id
-   \param layer layer number for thematic mapping
-   \param color color column name
-   \param width width column name
-   \param colors pointer to Colors structure or NULL
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GV_set_style_thematic(int id, int layer, const char* color, const char* width,
-			  struct Colors *color_rules)
-{
-    geovect *gv;
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return -1;
-    }
-
-    if(!gv->tstyle)
-	gv->tstyle = (gvstyle_thematic *)G_malloc(sizeof(gvstyle_thematic));
-    G_zero(gv->tstyle, sizeof(gvstyle_thematic));
-    
-    gv->tstyle->active = 1;
-    gv->tstyle->layer = layer;
-    if (color)
-	gv->tstyle->color_column = G_store(color);
-    if (width)
-	gv->tstyle->width_column = G_store(width);
-
-    Gv_load_vect_thematic(gv, color_rules);
-
-    return 1;
-}
-
-/*!
-   \brief Make style for thematic mapping inactive
-   
-   \param id vector set id
-
-   \return 1 on success
-   \return -1 on error (point set not found)
- */
-int GV_unset_style_thematic(int id)
-{
-    geovect *gv;
-
-    G_debug(4, "GV_unset_style_thematic(): id=%d", id);
-
-    if (NULL == (gv = gv_get_vect(id))) {
-	return -1;
-    }
-
-    if (gv->tstyle) {
-	gv->tstyle->active = 0;
-    }
-
-    return 1;
-}
-
-/*!
-   \brief Set trans ?
-
-   \param id vector set id
-   \param xtrans,ytrans,ztrans x/y/z trans values
- */
-void GV_set_trans(int id, float xtrans, float ytrans, float ztrans)
-{
-    geovect *gv;
-
-    G_debug(3, "GV_set_trans");
-
-    gv = gv_get_vect(id);
-
-    if (gv) {
-	gv->x_trans = xtrans;
-	gv->y_trans = ytrans;
-	gv->z_trans = ztrans;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get trans ?
-
-   \param id vector set id
-   \param[out] xtrans,ytrans,ztrans x/y/z trans values
- */
-int GV_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
-{
-    geovect *gv;
-
-    gv = gv_get_vect(id);
-
-    if (gv) {
-	*xtrans = gv->x_trans;
-	*ytrans = gv->y_trans;
-	*ztrans = gv->z_trans;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Select surface identified by hs to have vector identified
-   by hv draped over it
-
-   \param hv vector set id
-   \param hs surface id
-
-   \return 1 on success
-   \return -1 on error
- */
-int GV_select_surf(int hv, int hs)
-{
-    geovect *gv;
-
-    if (GV_surf_is_selected(hv, hs)) {
-	return (1);
-    }
-
-    gv = gv_get_vect(hv);
-
-    if (gv && GS_surf_exists(hs)) {
-	gv->drape_surf_id[gv->n_surfs] = hs;
-	gv->n_surfs += 1;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Unselect surface
-
-   \param hv vector set id
-   \param hs surface id
-
-   \return 1 on success
-   \return -1 on error
- */
-int GV_unselect_surf(int hv, int hs)
-{
-    geovect *gv;
-    int i, j;
-
-    if (!GV_surf_is_selected(hv, hs)) {
-	return (1);
-    }
-
-    gv = gv_get_vect(hv);
-
-    if (gv) {
-	for (i = 0; i < gv->n_surfs; i++) {
-	    if (gv->drape_surf_id[i] == hs) {
-		for (j = i; j < gv->n_surfs - 1; j++) {
-		    gv->drape_surf_id[j] = gv->drape_surf_id[j + 1];
-		}
-
-		gv->n_surfs -= 1;
-
-		return (1);
-	    }
-	}
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Check if surface is selected
-
-   \param hv vector set id
-   \param hs surface id
-
-   \return 1 selected
-   \return 0 not selected
- */
-int GV_surf_is_selected(int hv, int hs)
-{
-    int i;
-    geovect *gv;
-
-    gv = gv_get_vect(hv);
-
-    if (gv) {
-	for (i = 0; i < gv->n_surfs; i++) {
-	    if (hs == gv->drape_surf_id[i]) {
-		return (1);
-	    }
-	}
-    }
-
-    return (0);
-}
-
-/*!
-   \brief Draw vector set
-
-   \param vid vector set id
- */
-void GV_draw_vect(int vid)
-{
-    geosurf *gs;
-    geovect *gv;
-    int i;
-
-    gv = gv_get_vect(vid);
-
-    if (gv) {
-	for (i = 0; i < gv->n_surfs; i++) {
-	    gs = gs_get_surf(gv->drape_surf_id[i]);
-
-	    if (gs) {
-		gvd_vect(gv, gs, 0);
-	    }
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all loaded vector sets
- */
-void GV_alldraw_vect(void)
-{
-    int id;
-
-    for (id = 0; id < Next_vect; id++) {
-	GV_draw_vect(Vect_ID[id]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw vector set (fast mode)
-
-   \todo Seems to be broken, nothing is drawn
-
-   \param vid vector set id
- */
-void GV_draw_fastvect(int vid)
-{
-    geosurf *gs;
-    geovect *gv;
-    int i;
-
-    gv = gv_get_vect(vid);
-
-    if (gv) {
-	for (i = 0; i < gv->n_surfs; i++) {
-	    gs = gs_get_surf(gv->drape_surf_id[i]);
-
-	    if (gs) {
-		gvd_vect(gv, gs, 1);
-	    }
-	}
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all loaded vector sets (fast mode)
- */
-void GV_alldraw_fastvect(void)
-{
-    int id;
-
-    for (id = 0; id < Next_vect; id++) {
-	GV_draw_fastvect(Vect_ID[id]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Set client data
-
-   \param id vector set id
-   \param clientd pointer to client data
-
-   \return 1 on success
-   \return -1 on error
- */
-int GV_Set_ClientData(int id, void *clientd)
-{
-    geovect *gv;
-
-    gv = gv_get_vect(id);
-    if (gv) {
-	gv->clientdata = clientd;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get client data
-
-   \param id vector set id
-
-   \return pointer to client data
-   \return NULL on error
- */
-void *GV_Get_ClientData(int id)
-{
-    geovect *gv;
-
-    gv = gv_get_vect(id);
-
-    if (gv) {
-	return (gv->clientdata);
-    }
-
-    return (NULL);
-}

Deleted: grass/trunk/lib/ogsf/GVL2.c
===================================================================
--- grass/trunk/lib/ogsf/GVL2.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/GVL2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,1521 +0,0 @@
-/*!
-   \file GVL2.c
-
-   \brief OGSF library - loading and manipulating volumes
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown UI-GMSL (May 1997)
-   Tomas Paudits (February 2004)
- */
-
-#include <string.h>
-#include <grass/gis.h>
-#include <grass/raster3d.h>
-#include <grass/ogsf.h>
-#include <grass/glocale.h>
-#include "gsget.h"
-
-static int Vol_ID[MAX_VOLS];
-static int Next_vol = 0;
-
-static RASTER3D_Region wind3;
-static double Region[6];
-
-/*!
-   \brief Library intialization for volumes
-
-   Set region extent (N,S,W,E,T,B)
- */
-void GVL_libinit(void)
-{
-    Rast3d_init_defaults();
-    Rast3d_get_window(&wind3);
-
-    Region[0] = wind3.north;
-    Region[1] = wind3.south;
-    Region[2] = wind3.west;
-    Region[3] = wind3.east;
-    Region[4] = wind3.top;
-    Region[5] = wind3.bottom;
-
-    return;
-}
-
-/*!
-   \brief Initialize 3D region
-
-   Set region extent (N,S,W,E,T,B)
- */
-void GVL_init_region(void)
-{
-    Rast3d_read_window(&wind3, NULL);
-
-    Region[0] = wind3.north;
-    Region[1] = wind3.south;
-    Region[2] = wind3.west;
-    Region[3] = wind3.east;
-    Region[4] = wind3.top;
-    Region[5] = wind3.bottom;
-
-    return;
-}
-
-/*!
-   \brief Get region extent settings
-
-   \param[out] n,s,w,e north, south, west, east
-   \param[out] t,b top, bottom
-
-   \return 1
- */
-int GVL_get_region(float *n, float *s, float *w, float *e, float *t, float *b)
-{
-    *n = Region[0];
-    *s = Region[1];
-    *w = Region[2];
-    *e = Region[3];
-    *t = Region[4];
-    *b = Region[5];
-
-    return (1);
-}
-
-/*!
-   \brief Get window 
-
-   \todo gvl_file.c use this - change
-
-   \return pointer to RASTER3D_Region struct (static)
- */
-void *GVL_get_window()
-{
-    return &wind3;
-}
-
-/*!
-   \brief Check if volume set exists
-
-   \param id volume set id
-
-   \return 1 found
-   \return 0 not found
- */
-int GVL_vol_exists(int id)
-{
-    int i, found = 0;
-
-    G_debug(3, "GVL_vol_exists");
-
-    if (NULL == gvl_get_vol(id)) {
-	return (0);
-    }
-
-    for (i = 0; i < Next_vol && !found; i++) {
-	if (Vol_ID[i] == id) {
-	    found = 1;
-	}
-    }
-
-    return (found);
-}
-
-/*!
-   \brief Create new volume set
-
-   \return volume set id
-   \return -1 on error
- */
-int GVL_new_vol(void)
-{
-    geovol *nvl;
-
-    G_debug(3, "GVL_new_vol():");
-
-    if (Next_vol < MAX_VOLS) {
-	nvl = gvl_get_new_vol();
-
-	gvl_init_vol(nvl, wind3.west + wind3.ew_res / 2.,
-		     wind3.south + wind3.ns_res / 2., wind3.bottom,
-		     wind3.rows, wind3.cols, wind3.depths,
-		     wind3.ew_res, wind3.ns_res, wind3.tb_res);
-
-	Vol_ID[Next_vol] = nvl->gvol_id;
-	++Next_vol;
-
-	G_debug(3, "    id=%d", nvl->gvol_id);
-	
-	return (nvl->gvol_id);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get number of loaded volume sets
-
-   \return number of volume sets
- */
-int GVL_num_vols(void)
-{
-    return (gvl_num_vols());
-}
-
-/*!
-   \brief Get list of loaded volume sets
-
-   Must be freed if not needed!
-
-   \param[out] numvols number of volume sets
-
-   \return pointer to list of volume sets
-   \return NULL on error
- */
-int *GVL_get_vol_list(int *numvols)
-{
-    int i, *ret;
-
-    *numvols = Next_vol;
-
-    if (Next_vol) {
-	ret = (int *)G_malloc(Next_vol * sizeof(int));
-	if (!ret)
-	    return (NULL);
-
-	for (i = 0; i < Next_vol; i++) {
-	    ret[i] = Vol_ID[i];
-	}
-
-	return (ret);
-    }
-
-    return (NULL);
-}
-
-/*!
-   \brief Delete volume set from list
-
-   \param id volume set id
-
-   \return 1 on success
-   \return -1 on error (invalid volume set id)
- */
-int GVL_delete_vol(int id)
-{
-    int i, j, found = 0;
-
-    G_debug(3, "GVL_delete_vol");
-
-    if (GVL_vol_exists(id)) {
-
-	for (i = 0; i < GVL_isosurf_num_isosurfs(id); i++) {
-	    GVL_isosurf_del(id, 0);
-	}
-
-	for (i = 0; i < GVL_slice_num_slices(id); i++) {
-	    GVL_slice_del(id, 0);
-	}
-
-	gvl_delete_vol(id);
-
-	for (i = 0; i < Next_vol && !found; i++) {
-	    if (Vol_ID[i] == id) {
-		found = 1;
-		for (j = i; j < Next_vol; j++) {
-		    Vol_ID[j] = Vol_ID[j + 1];
-		}
-	    }
-	}
-
-	if (found) {
-	    --Next_vol;
-
-	    return (1);
-	}
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Load 3d raster map to volume set
-
-   \param id volume set id
-   \param filename 3d raster map name
-
-   \return -1 on error
-   \return 0 on success
- */
-int GVL_load_vol(int id, const char *filename)
-{
-    geovol *gvl;
-    int handle;
-
-    G_debug(3, "GVL_load_vol(): id=%d, name=%s", id, filename);
-
-    if (NULL == (gvl = gvl_get_vol(id))) {
-	return (-1);
-    }
-
-    G_message(_("Loading 3d raster map <%s>..."), filename);
-
-    if (0 > (handle = gvl_file_newh(filename, VOL_FTYPE_RASTER3D)))
-	return (-1);
-
-    gvl->hfile = handle;
-
-    return (0);
-}
-
-/*!
-   \brief Get volume set name
-
-   \param id volume set id
-   \param[out] filename name (must be allocated)
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_get_volname(int id, char *filename)
-{
-    geovol *gvl;
-
-    if (NULL == (gvl = gvl_get_vol(id))) {
-	return (-1);
-    }
-
-    if (0 > gvl->hfile) {
-	return (-1);
-    }
-
-    strcpy(filename, gvl_file_get_name(gvl->hfile));
-
-    return (1);
-}
-
-/*!
-   \brief Get volume dimensions
-
-   \param id volume set id
-   \param[out] rows,cols,depths number of rows, cols, depths
- */
-void GVL_get_dims(int id, int *rows, int *cols, int *depths)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*rows = gvl->rows;
-	*cols = gvl->cols;
-	*depths = gvl->depths;
-    }
-
-    G_debug(3, "GVL_get_dims() id=%d, rows=%d, cols=%d, depths=%d",
-	    gvl->gvol_id, gvl->rows, gvl->cols, gvl->depths);
-    
-    return;
-}
-
-/*!
-   \brief Set trans ?
-
-   \param id volume set id
-   \param xtrans,ytrans,ztrans x/y/z trans values
- */
-void GVL_set_trans(int id, float xtrans, float ytrans, float ztrans)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_set_trans");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->x_trans = xtrans;
-	gvl->y_trans = ytrans;
-	gvl->z_trans = ztrans;
-    }
-
-    return;
-}
-
-/*!
-   \brief Get trans ?
-
-   \param id volume set id
-   \param[out] xtrans,ytrans,ztrans x/y/z trans values
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*xtrans = gvl->x_trans;
-	*ytrans = gvl->y_trans;
-	*ztrans = gvl->z_trans;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set drawing wire box
-
-   \param id volume set id
-   \param draw_wire 1 for drawing wire, 0 otherwise
- */
-void GVL_set_draw_wire(int id, int draw_wire)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_set_draw_wire");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->draw_wire = draw_wire;
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw volume set
-
-   \param vid volume set id
- */
-void GVL_draw_vol(int vid)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(vid);
-
-    if (gvl) {
-	gvld_vol(gvl);
-        if (gvl->draw_wire) {
-	    gvld_wind3_box(gvl);
-        }
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw volume in wire mode
-
-   \param id volume set id
- */
-void GVL_draw_wire(int id)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_draw_wire(): id=%d", id);
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvld_wire_vol(gvl);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all volume sets
- */
-void GVL_alldraw_vol(void)
-{
-    int id;
-
-    for (id = 0; id < Next_vol; id++) {
-	GVL_draw_vol(Vol_ID[id]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Draw all volume sets in wire mode
- */
-void GVL_alldraw_wire(void)
-{
-    int id;
-
-    for (id = 0; id < Next_vol; id++) {
-	GVL_draw_wire(Vol_ID[id]);
-    }
-
-    return;
-}
-
-/*!
-   \brief Set client data for volume set
-
-   \param id volume set id
-   \param clientd pointer to client data
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_Set_ClientData(int id, void *clientd)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->clientdata = clientd;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get client data
-
-   \param id volume set id
-
-   \return pointer to client data
-   \return NULL on error
- */
-void *GVL_Get_ClientData(int id)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	return (gvl->clientdata);
-    }
-
-    return (NULL);
-}
-
-/*!
-   \brief Set focus on map center
-
-   \param id volume set id
- */
-void GVL_set_focus_center_map(int id)
-{
-    float center[3];
-    geovol *gvl;
-
-    G_debug(3, "GS_set_focus_center_map");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	center[X] = (gvl->xmax - gvl->xmin) / 2.;
-	center[Y] = (gvl->ymax - gvl->ymin) / 2.;
-	center[Z] = (gvl->zmax - gvl->zmin) / 2.;
-
-	GS_set_focus(center);
-    }
-
-    return;
-}
-
-/************************************************************************/
-/* ISOSURFACES */
-
-/************************************************************************/
-
-/*!
-   \brief Get draw resolution for isosurface
-
-   \todo error handling
-
-   \param id volume set id
-   \param[out] xres,yres,zres x/y/z resolution value
- */
-void GVL_isosurf_get_drawres(int id, int *xres, int *yres, int *zres)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_isosurf_get_drawres");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*xres = gvl->isosurf_x_mod;
-	*yres = gvl->isosurf_y_mod;
-	*zres = gvl->isosurf_z_mod;
-    }
-
-    return;
-}
-
-/*!
-   \brief Set isosurface draw resolution
-
-   \param id volume set id
-   \param xres,yres,zres x/y/z resolution value
-
-   \return -1 on error (invalid values/volume set id)
-   \return 0 on success
- */
-int GVL_isosurf_set_drawres(int id, int xres, int yres, int zres)
-{
-    geovol *gvl;
-    int i;
-
-    G_debug(3, "GVL_isosurf_set_drawres(): id=%d", id);
-
-    if (xres < 1 || yres < 1 || zres < 1) {
-	return (-1);
-    }
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->isosurf_x_mod = xres;
-	gvl->isosurf_y_mod = yres;
-	gvl->isosurf_z_mod = zres;
-
-	for (i = 0; i < gvl->n_isosurfs; i++) {
-	    gvl_isosurf_set_att_changed(gvl->isosurf[i], ATT_TOPO);
-	}
-
-	return (0);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get isosurface draw mode
-
-   \param id volume set id
-   \param[out] mode draw-mode
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_get_drawmode(int id, int *mode)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*mode = gvl->isosurf_draw_mode;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set isosurface draw mode
-
-   \param id volume set id
-   \param mode draw mode
-
-   \return 0 on success
-   \return -1 on error (invalid volume set id)
- */
-int GVL_isosurf_set_drawmode(int id, int mode)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_isosurf_set_drawmode(): id=%d mode=%d", id, mode);
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->isosurf_draw_mode = mode;
-
-	return (0);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Add isosurface
-
-   \param id volume set id
-
-   \return -1 on error (invalid volume set id
-   \return 1 on success
- */
-int GVL_isosurf_add(int id)
-{
-    geovol *gvl;
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_add() id=%d", id);
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (gvl->n_isosurfs == MAX_ISOSURFS)
-	return (-1);
-
-    isosurf = (geovol_isosurf *) G_malloc(sizeof(geovol_isosurf));
-    if (!isosurf) {
-	return (-1);
-    }
-
-    gvl_isosurf_init(isosurf);
-
-    gvl->n_isosurfs++;
-    gvl->isosurf[gvl->n_isosurfs - 1] = (geovol_isosurf *) isosurf;
-
-    return (1);
-}
-
-/*!
-   \brief Delete isosurface
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_isosurf_del(int id, int isosurf_id)
-{
-    geovol *gvl;
-    geovol_isosurf *isosurf;
-    int i;
-
-    G_debug(3, "GVL_isosurf_del");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (!isosurf)
-	return (-1);
-
-    if (!gvl_isosurf_freemem(isosurf)) {
-	return (-1);
-    }
-
-    gvl = gvl_get_vol(id);
-
-    G_free(gvl->isosurf[isosurf_id]);
-
-    for (i = isosurf_id + 1; i < gvl->n_isosurfs; i++) {
-	gvl->isosurf[i - 1] = gvl->isosurf[i];
-    }
-
-    gvl->n_isosurfs--;
-
-    return (1);
-}
-
-/*!
-   \brief Move up isosurface in list
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_isosurf_move_up(int id, int isosurf_id)
-{
-    geovol *gvl;
-    geovol_isosurf *tmp;
-
-    G_debug(3, "GVL_isosurf_move_up");
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (isosurf_id < 0 || isosurf_id > (gvl->n_isosurfs - 1))
-	return (-1);
-
-    if (isosurf_id == 0)
-	return (1);
-
-    tmp = gvl->isosurf[isosurf_id - 1];
-    gvl->isosurf[isosurf_id - 1] = gvl->isosurf[isosurf_id];
-    gvl->isosurf[isosurf_id] = tmp;
-
-    return (1);
-}
-
-/*!
-   \brief Move down isosurface in list
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_isosurf_move_down(int id, int isosurf_id)
-{
-    geovol *gvl;
-    geovol_isosurf *tmp;
-
-    G_debug(3, "GVL_isosurf_move_up");
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (isosurf_id < 0 || isosurf_id > (gvl->n_isosurfs - 1))
-	return (-1);
-
-    if (isosurf_id == (gvl->n_isosurfs - 1))
-	return (1);
-
-    tmp = gvl->isosurf[isosurf_id + 1];
-    gvl->isosurf[isosurf_id + 1] = gvl->isosurf[isosurf_id];
-    gvl->isosurf[isosurf_id] = tmp;
-
-    return (1);
-}
-
-/*!
-   \brief Get isosurface attributes
-
-   \param id volume set id
-   \param isosurf_id surface id
-   \param att attribute id
-   \param[out] set
-   \param[out] constant
-   \param[out] mapname
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_isosurf_get_att(int id, int isosurf_id,
-			int att, int *set, float *constant, char *mapname)
-{
-    int src;
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_get_att");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	if (-1 != (src = gvl_isosurf_get_att_src(isosurf, att))) {
-	    *set = src;
-
-	    if (src == CONST_ATT) {
-		*constant = isosurf->att[att].constant;
-	    }
-	    else if (src == MAP_ATT) {
-		strcpy(mapname, gvl_file_get_name(isosurf->att[att].hfile));
-	    }
-
-	    return (1);
-	}
-
-	return (-1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Unset isosurface attributes
-
-   \param id volume set id
-   \param isosurface_id isosurface id
-   \param att attribute id
-
-   \return ?
-   \return -1 on error
- */
-int GVL_isosurf_unset_att(int id, int isosurf_id, int att)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_unset_att");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	return (gvl_isosurf_set_att_src(isosurf, att, NOTSET_ATT));
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set constant isosurface attribute
-
-   Attributes:
-    - ATT_NORM
-    - ATT_TOPO topography (level) constant
-    - ATT_COLOR color map/constant
-    - ATT_MASK mask map
-    - ATT_TRANSP transparency map/constant
-    - ATT_SHINE shininess map/constant
-    - ATT_EMIT emission map/constant
-
-   \param id volume set id
-   \param isosurf_id isosurface id (0 - MAX_ISOSURFS)
-   \param att attribute descriptor
-   \param constant constant value
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_set_att_const(int id, int isosurf_id, int att, float constant)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_set_att_const() id=%d isosurf_id=%d "
-	    "att=%d const=%f", id, isosurf_id, att, constant);
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	return (gvl_isosurf_set_att_const(isosurf, att, constant));
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set isosurface map attribute
-
-   Attributes:
-    - ATT_NORM
-    - ATT_TOPO topography (level) constant
-    - ATT_COLOR color map/constant
-    - ATT_MASK mask map
-    - ATT_TRANSP transparency map/constant
-    - ATT_SHINE shininess map/constant
-    - ATT_EMIT emission map/constant
-
-   \param id volume set id
-   \param isosurf_id isosurface id (0 - MAX_ISOSURFS)
-   \param att attribute descriptor
-   \param filename map name
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_set_att_map(int id, int isosurf_id, int att,
-			    const char *filename)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_set_att_map(): id=%d, isosurf_id=%d "
-	    "att=%d map=%s", id, isosurf_id, att, filename);
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	return gvl_isosurf_set_att_map(isosurf, att, filename);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get isosurface flags
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-   \param[out] inout map name
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_get_flags(int id, int isosurf_id, int *inout)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_get_flags");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	*inout = isosurf->inout_mode;
-
-	return (1);
-    }
-    return (-1);
-}
-
-/*!
-   \brief Set isosurface flags
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-   \param inout map name
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_set_flags(int id, int isosurf_id, int inout)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_get_flags");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	isosurf->inout_mode = inout;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get number of available isosurfaces
-
-   \param id volume set id
-
-   \return number of isosurfaces
-   \return -1 on error
- */
-int GVL_isosurf_num_isosurfs(int id)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_isosurf_num_isosurfs");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	return gvl->n_isosurfs;
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set mask attribute mode
-
-   Mask attribute special: constant is set to indicate invert or no
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-   \param mode attribute mode
-
-   \return mode id
-   \return -1 on error
- */
-int GVL_isosurf_set_maskmode(int id, int isosurf_id, int mode)
-{
-    geovol_isosurf *isosurf;
-
-    G_debug(3, "GVL_isosurf_set_att_const");
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	isosurf->att[ATT_MASK].constant = mode;
-
-	return (mode);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get isosurface mask mode
-
-   \param id volume set id
-   \param isosurf_id isosurface id
-   \param mode attribute mode
-
-   \return 1 on success
-   \return -1 on error
- */
-int GVL_isosurf_get_maskmode(int id, int isosurf_id, int *mode)
-{
-    geovol_isosurf *isosurf;
-
-    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
-
-    if (isosurf) {
-	*mode = isosurf->att[ATT_MASK].constant;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/************************************************************************/
-/* SLICES */
-
-/************************************************************************/
-
-/*!
-   \brief Get draw resolution of slice
-
-   \param id volume set id
-   \param[out] xres,yres,zres x/y/z resolution value
- */
-void GVL_slice_get_drawres(int id, int *xres, int *yres, int *zres)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_slice_get_drawres");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*xres = gvl->slice_x_mod;
-	*yres = gvl->slice_y_mod;
-	*zres = gvl->slice_z_mod;
-    }
-
-    return;
-}
-
-/*!
-   \brief Set slice draw resolution
-
-   \param id volume set id
-   \param xres,yres,zres x/y/z resolution value
-
-   \return 0 on success
-   \return -1 on error (invalid value or id)
- */
-int GVL_slice_set_drawres(int id, int xres, int yres, int zres)
-{
-    geovol *gvl;
-    int i;
-
-    G_debug(3, "GVL_slice_set_drawres(): id=%d", id);
-
-    if (xres < 1 || yres < 1 || zres < 1) {
-	return (-1);
-    }
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->slice_x_mod = xres;
-	gvl->slice_y_mod = yres;
-	gvl->slice_z_mod = zres;
-
-	for (i = 0; i < gvl->n_slices; i++) {
-	    gvl->slice[i]->changed = 1;
-	}
-
-	return (0);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get slice draw mode
-
-   \param id volume set id
-   \param[out] mode draw mode
-
-   \return 1 on success
-   \return -1 on error (invalid id)
- */
-int GVL_slice_get_drawmode(int id, int *mode)
-{
-    geovol *gvl;
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	*mode = gvl->slice_draw_mode;
-
-	return (1);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Set slice draw mode
-
-   \param id volume set id
-   \param mode draw mode
-
-   \return 0 on success
-   \return -1 on error (invalid id)
- */
-int GVL_slice_set_drawmode(int id, int mode)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_slice_set_drawmode(): id=%d, mode=%d", id, mode);
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	gvl->slice_draw_mode = mode;
-
-	return (0);
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Add slice
-
-   \param id volume set id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_add(int id)
-{
-    geovol *gvl;
-    geovol_slice *slice;
-
-    G_debug(3, "GVL_slice_add");
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (gvl->n_slices == MAX_SLICES)
-	return (-1);
-
-    if (NULL == (slice = (geovol_slice *) G_malloc(sizeof(geovol_slice)))) {
-	return (-1);
-    }
-
-    gvl_slice_init(slice);
-
-    gvl->n_slices++;
-    gvl->slice[gvl->n_slices - 1] = (geovol_slice *) slice;
-
-    return (1);
-}
-
-/*!
-   \brief Delete slice
-
-   \param id volume set id
-   \param slice_id slice id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_del(int id, int slice_id)
-{
-    geovol *gvl;
-    geovol_slice *slice;
-    int i;
-
-    G_debug(3, "GVL_slice_del");
-
-    slice = gvl_slice_get_slice(id, slice_id);
-
-    if (!slice)
-	return (-1);
-
-    if (!gvl_slice_freemem(slice)) {
-	return (-1);
-    }
-
-    gvl = gvl_get_vol(id);
-
-    G_free(gvl->slice[slice_id]);
-
-    for (i = slice_id + 1; i < gvl->n_slices; i++) {
-	gvl->slice[i - 1] = gvl->slice[i];
-    }
-
-    gvl->n_slices--;
-
-    return (1);
-}
-
-/*!
-   \brief Move up slice
-
-   \param id volume set id
-   \param slice_id slice id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_move_up(int id, int slice_id)
-{
-    geovol *gvl;
-    geovol_slice *tmp;
-
-    G_debug(3, "GVL_slice_move_up");
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (slice_id < 0 || slice_id > (gvl->n_slices - 1))
-	return (-1);
-
-    if (slice_id == 0)
-	return (1);
-
-    tmp = gvl->slice[slice_id - 1];
-    gvl->slice[slice_id - 1] = gvl->slice[slice_id];
-    gvl->slice[slice_id] = tmp;
-
-    return (1);
-}
-
-/*!
-   \brief Move down slice
-
-   \param id volume set id
-   \param slice_id slice id
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_move_down(int id, int slice_id)
-{
-    geovol *gvl;
-    geovol_slice *tmp;
-
-    G_debug(3, "GVL_slice_move_up");
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    if (slice_id < 0 || slice_id > (gvl->n_slices - 1))
-	return (-1);
-
-    if (slice_id == (gvl->n_slices - 1))
-	return (1);
-
-    tmp = gvl->slice[slice_id + 1];
-    gvl->slice[slice_id + 1] = gvl->slice[slice_id];
-    gvl->slice[slice_id] = tmp;
-
-    return (1);
-}
-
-/*!
-   \brief Get number or slices
-
-   \param id volume set id
-
-   \return number of slices
-   \return -1 on error
- */
-int GVL_slice_num_slices(int id)
-{
-    geovol *gvl;
-
-    G_debug(3, "GVL_isosurf_num_isosurfs");
-
-    gvl = gvl_get_vol(id);
-
-    if (gvl) {
-	return gvl->n_slices;
-    }
-
-    return (-1);
-}
-
-/*!
-   \brief Get slice position
-
-   \param id volume set id
-   \param slice_id slice id
-   \param[out] x1,y1,z1 coordinates ?
-   \param[out] x2,y2,z2 coordinates ?
-   \param[out] dir direction
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_get_pos(int id, int slice_id,
-		      float *x1, float *x2, float *y1, float *y2, float *z1,
-		      float *z2, int *dir)
-{
-    geovol *gvl;
-    geovol_slice *slice;
-    int cols, rows, depths;
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    slice = gvl_slice_get_slice(id, slice_id);
-
-    if (!slice)
-	return (-1);
-
-    if (slice->dir == X) {
-	cols = gvl->rows;
-	rows = gvl->depths;
-	depths = gvl->cols;
-    }
-    else if (slice->dir == Y) {
-	cols = gvl->cols;
-	rows = gvl->depths;
-	depths = gvl->rows;
-    }
-    else if (slice->dir == Z) {
-	cols = gvl->cols;
-	rows = gvl->rows;
-	depths = gvl->depths;
-    }
-    else {
-	return (-1);
-    }
-
-    *x1 = slice->x1 / (cols - 1);
-    *x2 = slice->x2 / (cols - 1);
-    *y1 = slice->y1 / (rows - 1);
-    *y2 = slice->y2 / (rows - 1);
-    *z1 = slice->z1 / (depths - 1);
-    *z2 = slice->z2 / (depths - 1);
-
-    *dir = slice->dir;
-
-    return (1);
-}
-
-/*!
-   \brief Get slice position
-
-   \param id volume set id
-   \param slice_id slice id
-   \param x1,y1,z1 coordinates ?
-   \param x2,y2,z2 coordinates ?
-   \param dir direction
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_set_pos(int id, int slice_id,
-		      float x1, float x2, float y1, float y2, float z1,
-		      float z2, int dir)
-{
-    geovol *gvl;
-    geovol_slice *slice;
-    int cols, rows, depths;
-
-    gvl = gvl_get_vol(id);
-
-    if (!gvl)
-	return (-1);
-
-    slice = gvl_slice_get_slice(id, slice_id);
-
-    if (!slice)
-	return (-1);
-
-    if (dir == X) {
-	cols = gvl->rows;
-	rows = gvl->depths;
-	depths = gvl->cols;
-    }
-    else if (dir == Y) {
-	cols = gvl->cols;
-	rows = gvl->depths;
-	depths = gvl->rows;
-    }
-    else if (dir == Z) {
-	cols = gvl->cols;
-	rows = gvl->rows;
-	depths = gvl->depths;
-    }
-    else {
-	return (-1);
-    }
-
-    slice->x1 = ((x1 < 0.) ? 0. : ((x1 > 1.) ? 1. : x1)) * (cols - 1);
-    slice->x2 = ((x2 < 0.) ? 0. : ((x2 > 1.) ? 1. : x2)) * (cols - 1);
-    slice->y1 = ((y1 < 0.) ? 0. : ((y1 > 1.) ? 1. : y1)) * (rows - 1);
-    slice->y2 = ((y2 < 0.) ? 0. : ((y2 > 1.) ? 1. : y2)) * (rows - 1);
-    slice->z1 = ((z1 < 0.) ? 0. : ((z1 > 1.) ? 1. : z1)) * (depths - 1);
-    slice->z2 = ((z2 < 0.) ? 0. : ((z2 > 1.) ? 1. : z2)) * (depths - 1);
-
-    slice->dir = dir;
-
-    slice->changed = 1;
-
-    return (1);
-}
-
-/*!
-   \brief Get slice trans ?
-
-   \param id volume set id
-   \param slice_id slice id
-   \param[out] transp transp value
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_get_transp(int id, int slice_id, int *transp)
-{
-    geovol_slice *slice;
-
-    G_debug(3, "GVL_get_transp");
-
-    slice = gvl_slice_get_slice(id, slice_id);
-
-    if (!slice)
-	return (-1);
-
-    *transp = slice->transp;
-
-    return (1);
-}
-
-/*!
-   \brief Set slice trans ?
-
-   \param id volume set id
-   \param slice_id slice id
-   \param transp transp value
-
-   \return -1 on error
-   \return 1 on success
- */
-int GVL_slice_set_transp(int id, int slice_id, int transp)
-{
-    geovol_slice *slice;
-
-    G_debug(3, "GVL_set_transp");
-
-    slice = gvl_slice_get_slice(id, slice_id);
-
-    if (!slice)
-	return (-1);
-
-    slice->transp = transp;
-
-    return (1);
-}

Deleted: grass/trunk/lib/ogsf/Gp3.c
===================================================================
--- grass/trunk/lib/ogsf/Gp3.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/Gp3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,292 +0,0 @@
-/*!
-   \file lib/ogsf/Gp3.c
-
-   \brief OGSF library - loading point sets (lower level functions)
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008, 2011 by the GRASS Development Team
-
-   This program is free software under the GNU General Public License
-   (>=v2).  Read the file COPYING that comes with GRASS for details.
-
-   \author Bill Brown USACERL, GMSL/University of Illinois (January 1994)
-   \author Updated by Martin Landa <landa.martin gmail.com>
-   (doxygenized in May 2008, thematic mapping in June 2011)
- */
-
-#include <stdlib.h>
-
-#include <grass/gis.h>
-#include <grass/colors.h>
-#include <grass/raster.h>
-#include <grass/vector.h>
-#include <grass/dbmi.h>
-#include <grass/glocale.h>
-#include <grass/ogsf.h>
-
-/*!
-   \brief Load to points to memory
-
-   The other alternative may be to load to a tmp file.
-
-   \param name name of vector map to be loaded
-   \param[out] nsites number of loaded points
-   \param[out] has_z 2D or 3D points data loaded?
-
-   \return pointer to geopoint struct (array)
-   \return NULL on failure
- */
-geopoint *Gp_load_sites(const char *name, int *nsites, int *has_z)
-{
-    struct Map_info map;
-    static struct line_pnts *Points = NULL;
-    struct line_cats *Cats = NULL;
-    geopoint *top, *gpt, *prev;
-    int np, ltype, eof;
-    struct Cell_head wind;
-    int ndim;
-    const char *mapset;
-
-    np = 0;
-    eof = 0;
-    
-    mapset = G_find_vector2(name, "");
-    if (!mapset) {
-	G_warning(_("Vector map <%s> not found"), name);
-	return NULL;
-    }
-    
-    Vect_set_open_level(1);
-    if (Vect_open_old(&map, name, "") == -1) {
-	G_fatal_error(_("Unable to open vector map <%s>"),
-		      G_fully_qualified_name(name, mapset));
-    }
-    
-    Points = Vect_new_line_struct();
-    Cats = Vect_new_cats_struct();
-    
-    top = gpt = (geopoint *) G_malloc(sizeof(geopoint));
-    G_zero(gpt, sizeof(geopoint));
-    if (!top) {
-	return NULL;
-    }
-
-    G_get_set_window(&wind);
-    Vect_set_constraint_region(&map, wind.north, wind.south, wind.east,
-			       wind.west, PORT_DOUBLE_MAX, -PORT_DOUBLE_MAX);
-
-    /* get ndim */
-    *has_z = 0;
-    ndim = 2;
-    if (Vect_is_3d(&map)) {
-	*has_z = 1;
-	ndim = 3;
-    }
-
-    while (eof == 0) {
-	ltype = Vect_read_next_line(&map, Points, Cats);
-	switch (ltype) {
-	case -1:
-	    {
-		G_warning(_("Unable to read vector map <%s>"),
-			  G_fully_qualified_name(name, mapset));
-		return NULL;
-	    }
-	case -2:		/* EOF */
-	    {
-		eof = 1;
-		continue;
-	    }
-	}
-	if ((ltype & GV_POINTS)) {
-	    np++;
-	    gpt->p3[X] = Points->x[0];
-	    gpt->p3[Y] = Points->y[0];
-
-	    if (ndim > 2) {
-		gpt->dims = 3;
-		gpt->p3[Z] = Points->z[0];
-	    }
-	    else {
-		gpt->dims = 2;
-	    }
-
-	    /* Store category info for thematic display */
-	    if (Cats->n_cats > 0) {
-		gpt->cats = Cats;
-		Cats = Vect_new_cats_struct();
-	    }
-	    else {
-		Vect_reset_cats(Cats);
-	    }
-	    /* initialize style */
-	    gpt->highlighted = 0;
-	    
-	    G_debug(5, "loading vector point %d x=%f y=%f ncats=%d",
-		    np, Points->x[0], Points->y[0], Cats->n_cats);
-
-	    gpt->next = (geopoint *) G_malloc(sizeof(geopoint));	/* G_fatal_error */
-	    G_zero(gpt->next, sizeof(geopoint));
-	    if (!gpt->next) {
-		return NULL;
-	    }
-
-	    prev = gpt;
-	    gpt = gpt->next;
-	}
-
-    }
-    if (np > 0) {
-	prev->next = NULL;
-	G_free(gpt);
-    }
-
-    Vect_close(&map);
-
-    if (!np) {
-	G_warning(_("No points from vector map <%s> fall within current region"),
-		  G_fully_qualified_name(name, mapset));
-	return (NULL);
-    }
-    else {
-	G_message(_("Vector map <%s> loaded (%d points)"),
-		  G_fully_qualified_name(name, mapset), np);
-    }
-
-    *nsites = np;
-
-    return top;
-}
-
-/*!
-  \brief Load styles for geopoints based on thematic mapping
-
-  \param gp pointer to geosite structure
-  \param colors pointer to Colors structure or NULL
-  
-  \return number of points defined by thematic mapping
-  \return -1 on error
-*/
-int Gp_load_sites_thematic(geosite *gp, struct Colors *colors)
-{
-    geopoint *gpt;
-
-    struct Map_info Map;
-    struct field_info *Fi;
-    
-    int nvals, cat, npts, nskipped;
-    int red, blu, grn;
-    const char *str;
-    const char *mapset;
-
-    dbDriver *driver;
-    dbValue value;
-    
-    if(!gp || !gp->tstyle || !gp->filename)
-	return -1;
-
-    mapset = G_find_vector2(gp->filename, "");
-    if (!mapset) {
-	G_fatal_error(_("Vector map <%s> not found"), gp->filename);
-    }
-    
-    Vect_set_open_level(1);
-    if (Vect_open_old(&Map, gp->filename, "") == -1) {
-	G_fatal_error(_("Unable to open vector map <%s>"),
-		      G_fully_qualified_name(gp->filename, mapset));
-    }
-    
-    Fi = Vect_get_field(&Map, gp->tstyle->layer);
-    if (!Fi) {
-	G_warning(_("Database connection not defined for layer %d"),
-		  gp->tstyle->layer);
-    }
-    else {
-	driver = db_start_driver_open_database(Fi->driver, Fi->database);
-	if (!driver)
-	    G_fatal_error(_("Unable to open database <%s> by driver <%s>"),
-			  Fi->database, Fi->driver);
-    }
-    G_message(_("Loading thematic points layer <%s>..."),
-	      G_fully_qualified_name(gp->filename, mapset));
-    npts = nskipped = 0;
-    for(gpt = gp->points; gpt; gpt = gpt->next) {
-	gpt->style = (gvstyle *) G_malloc(sizeof(gvstyle));
-	G_zero(gpt->style, sizeof(gvstyle));
-	
-	/* use default style */
-	gpt->style->color  = gp->style->color;
-	gpt->style->symbol = gp->style->symbol;
-	gpt->style->size   = gp->style->size;
-	gpt->style->width  = gp->style->width;
-	
-	cat = -1;
-	if (gpt->cats)
-	    Vect_cat_get(gpt->cats, gp->tstyle->layer, &cat);
-	if (cat < 0) {
-	    nskipped++;
-	    continue;
-	}
-
-	/* color */
-	if (colors) {
-	    if (!Rast_get_c_color((const CELL *) &cat, &red, &grn, &blu, colors)) {
-		G_warning(_("No color rule defined for category %d"), cat);
-		gpt->style->color = gp->style->color;
-	    }
-	    gpt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
-		((int)((blu) << 16) & BLU_MASK);
-	}
-	if (gp->tstyle->color_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->color_column, &value);
-	    if (nvals < 1)
-		continue;
-	    str = db_get_value_string(&value);
-	    if (!str)
-		continue;
-	    if (G_str_to_color(str, &red, &grn, &blu) != 1) {
-		G_warning(_("Invalid color definition (%s)"),
-			  str);
-		gpt->style->color = gp->style->color;
-	    }
-	    else {
-		gpt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
-		    ((int)((blu) << 16) & BLU_MASK);
-	    }
-	}
-	
-	/* size */
-	if (gp->tstyle->size_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->size_column, &value);
-	    if (nvals < 1)
-		continue;
-	    gpt->style->size = db_get_value_int(&value);
-	}
-
-	/* width */
-	if (gp->tstyle->width_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->width_column, &value);
-	    if (nvals < 1)
-		continue;
-	    gpt->style->width = db_get_value_int(&value);
-	}
-
-	/* symbol/marker */
-	if (gp->tstyle->symbol_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->symbol_column, &value);
-	    if (nvals < 1)
-		continue;
-	    str = db_get_value_string(&value);
-	    gpt->style->symbol = GP_str_to_marker(str);
-	}
-	
-	npts++;
-    }
-    
-    if (nskipped > 0)
-	G_warning(_("%d points without category. "
-		    "Unable to determine color rules for features without category."),
-		  nskipped);
-    return npts;
-}

Deleted: grass/trunk/lib/ogsf/Gs3.c
===================================================================
--- grass/trunk/lib/ogsf/Gs3.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/Gs3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,1198 +0,0 @@
-/*!
-   \file Gs3.c
-
-   \brief OGSF library - loading surfaces (lower level functions)
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Bill Brown USACERL, GMSL/University of Illinois (January 1993)
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <stdlib.h>
-#include <string.h>
-
-#include <grass/gis.h>
-#include <grass/raster.h>
-#include <grass/glocale.h>
-#include <grass/bitmap.h>
-
-#include <grass/ogsf.h>
-/* for geoview & geodisplay in 3dview stuff */
-#include "gsget.h"
-/* for update_attrange - might be able to move this func now */
-
-/*!
-   \brief Used in the function Gs_update_attrange()
- */
-#define INIT_MINMAX(p, nm, size, min, max, found) \
-	found = 0; \
-	p+=(size-1); \
-	while (size--) \
-	{ \
-	    if (!BM_GET_BYOFFSET(nm, size)) \
-	    { \
-		min = max = *p; \
-		found = 1; \
-		break; \
-	    } \
-	    p--; \
-	}
-
-/*!
-   \brief Used in the function Gs_update_attrange()
- */
-#define SET_MINMAX(p, nm, size, min, max) \
-	p+=(size-1); \
-	while(size--) \
-	{ \
-	    if (!BM_GET_BYOFFSET(nm, size)) \
-	    { \
-		if (*p < min) \
-		{ \
-		    min = *p; \
-		} \
-		else if (*p > max) \
-		{ \
-		    max = *p; \
-		}  \
-	    } \
-	    p--; \
-	}
-
-typedef int FILEDESC;
-
-#define NO_DATA_COL 0xffffff
-
-/*!
-   \brief Calculates distance in METERS between two points in current projection (2D)
-
-   Uses G_distance().
-
-   \param from 'from' point (X, Y)
-   \param to 'to' point (X, Y)
-
-   \return distance
- */
-double Gs_distance(double *from, double *to)
-{
-    static int first = 1;
-
-    if (first) {
-	first = 0;
-	G_begin_distance_calculations();
-    }
-
-    return G_distance(from[0], from[1], to[0], to[1]);
-}
-
-/*!
-   \brief Load raster map as floating point map
-
-   Calling function must have already allocated space in buff for
-   wind->rows * wind->cols floats.
-
-   This routine simply loads the map into a 2d array by repetitve calls
-   to get_f_raster_row.
-
-   \param wind current window
-   \param map_name raster map name
-   \param[out] buff data buffer
-   \param[out] nullmap null map buffer
-   \param[out] has_null indicates if raster map contains null-data
-
-   \return 1 on success
-   \return 0 on failure
- */
-int Gs_loadmap_as_float(struct Cell_head *wind, const char *map_name,
-			float *buff, struct BM *nullmap, int *has_null)
-{
-    FILEDESC cellfile;
-    const char *map_set;
-    int offset, row, col;
-
-    G_debug(3, "Gs_loadmap_as_float(): name=%s", map_name);
-
-    map_set = G_find_raster2(map_name, "");
-    if (!map_set) {
-	G_warning(_("Raster map <%s> not found"), map_name);
-	return 0;
-    }
-    *has_null = 0;
-
-    cellfile = Rast_open_old(map_name, map_set);
-
-    G_message(_("Loading raster map <%s>..."),
-	      G_fully_qualified_name(map_name, map_set));
-
-    for (row = 0; row < wind->rows; row++) {
-	offset = row * wind->cols;
-	Rast_get_f_row(cellfile, &(buff[offset]), row);
-
-	G_percent(row, wind->rows, 2);
-
-	for (col = 0; col < wind->cols; col++) {
-	    if (Rast_is_f_null_value(buff + offset + col)) {
-		*has_null = 1;
-		BM_set(nullmap, col, row, 1);
-	    }
-	    /* set nm */
-	}
-    }
-    G_percent(1, 1, 1);
-
-    G_debug(4, "  has_null=%d", *has_null);
-
-    Rast_close(cellfile);
-
-    return (1);
-}
-
-/*!
-   \brief Load raster map as integer map
-
-   Calling function must have already allocated space in buff for
-   wind->rows * wind->cols floats.
-
-   This routine simply loads the map into a 2d array by repetitve calls
-   to get_f_raster_row.
-
-   \todo fn body of Gs_loadmap_as_float()
-
-   \param wind current window
-   \param map_name raster map name
-   \param[out] buff data buffer
-   \param[out] nullmap null map buffer
-   \param[out] has_null indicates if raster map contains null-data
-
-   \return 1 on success
-   \return 0 on failure
- */
-int Gs_loadmap_as_int(struct Cell_head *wind, const char *map_name, int *buff,
-		      struct BM *nullmap, int *has_null)
-{
-    FILEDESC cellfile;
-    const char *map_set;
-    int offset, row, col;
-
-    G_debug(3, "Gs_loadmap_as_int");
-
-    map_set = G_find_raster2(map_name, "");
-    if (!map_set) {
-	G_warning(_("Raster map <%s> not found"), map_name);
-	return 0;
-    }
-    *has_null = 0;
-
-    cellfile = Rast_open_old(map_name, map_set);
-
-    G_message(_("Loading raster map <%s>..."),
-	      G_fully_qualified_name(map_name, map_set));
-
-    for (row = 0; row < wind->rows; row++) {
-	offset = row * wind->cols;
-	Rast_get_c_row(cellfile, &(buff[offset]), row);
-
-	G_percent(row, wind->rows, 2);
-
-	for (col = 0; col < wind->cols; col++) {
-	    if (Rast_is_f_null_value(buff + offset + col)) {
-		*has_null = 1;
-		BM_set(nullmap, col, row, 1);
-	    }
-
-	    /* set nm */
-	}
-    }
-    G_percent(1, 1, 1);
-    
-    Rast_close(cellfile);
-
-    return (1);
-}
-
-/*!
-   \brief Get map data type
-
-   \param filename raster map name
-   \param negflag
-
-   \return -1 if map is integer and Rast_read_range() fails
-   \return data type (ARRY_*)
- */
-int Gs_numtype(const char *filename, int *negflag)
-{
-    CELL max = 0, min = 0;
-    struct Range range;
-    const char *mapset;
-    int shortbits, charbits, bitplace;
-    static int max_short, max_char;
-    static int first = 1;
-
-    if (first) {
-	max_short = max_char = 1;
-	shortbits = 8 * sizeof(short);
-
-	for (bitplace = 1; bitplace < shortbits; ++bitplace) {
-	    /*1 bit for sign */
-	    max_short *= 2;
-	}
-
-	max_short -= 1;
-
-	/* NO bits for sign, using unsigned char */
-	charbits = 8 * sizeof(unsigned char);
-
-	for (bitplace = 0; bitplace < charbits; ++bitplace) {
-	    max_char *= 2;
-	}
-
-	max_char -= 1;
-
-	first = 0;
-    }
-
-    mapset = G_find_raster2(filename, "");
-    if (!mapset) {
-	G_warning(_("Raster map <%s> not found"), filename);
-	return -1;
-    }
-
-    if (Rast_map_is_fp(filename, mapset)) {
-	G_debug(3, "Gs_numtype(): fp map detected");
-
-	return (ATTY_FLOAT);
-    }
-
-    if (-1 == Rast_read_range(filename, mapset, &range)) {
-	return (-1);
-    }
-
-    Rast_get_range_min_max(&range, &min, &max);
-    *negflag = (min < 0);
-
-    if (max < max_char && min > 0) {
-	return (ATTY_CHAR);
-    }
-
-    if (max < max_short && min > -max_short) {
-	return (ATTY_SHORT);
-    }
-
-    return (ATTY_INT);
-}
-
-/*!
-   \brief Load raster map as integer map
-
-   Calling function must have already allocated space in buff for
-   wind->rows * wind->cols shorts.  
-
-   This routine simply loads the map into a 2d array by repetitve calls
-   to get_map_row.
-
-   \param wind current window
-   \param map_name raster map name
-   \param[out] buff data buffer
-   \param[out] nullmap null map buffer
-   \param[out] has_null indicates if raster map contains null-data
-
-   \return 1 on success
-   \return -1 on failure,
-   \return -2 if read ok, but 1 or more values were too large (small)
-   to fit into a short (in which case the max (min) short is used)
- */
-int Gs_loadmap_as_short(struct Cell_head *wind, const char *map_name,
-			short *buff, struct BM *nullmap, int *has_null)
-{
-    FILEDESC cellfile;
-    const char *map_set;
-    int *ti, *tmp_buf;
-    int offset, row, col, val, max_short, overflow, shortsize, bitplace;
-    short *ts;
-
-    G_debug(3, "Gs_loadmap_as_short");
-
-    overflow = 0;
-    shortsize = 8 * sizeof(short);
-
-    /* 1 bit for sign */
-    /* same as 2 << (shortsize-1) */
-    for (max_short = bitplace = 1; bitplace < shortsize; ++bitplace) {
-	max_short *= 2;
-    }
-
-    max_short -= 1;
-
-    map_set = G_find_raster2(map_name, "");
-    if (!map_set) {
-	G_warning(_("Raster map <%s> not found"), map_name);
-	return -1;
-    }
-    *has_null = 0;
-
-    cellfile = Rast_open_old(map_name, map_set);
-
-    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
-    if (!tmp_buf) {
-	return -1;
-    }
-
-    G_message(_("Loading raster map <%s>..."),
-	      G_fully_qualified_name(map_name, map_set));
-
-    for (row = 0; row < wind->rows; row++) {
-	offset = row * wind->cols;
-	Rast_get_c_row(cellfile, tmp_buf, row);
-
-	G_percent(row, wind->rows, 2);
-
-	ts = &(buff[offset]);
-	ti = tmp_buf;
-
-	for (col = 0; col < wind->cols; col++) {
-	    if (Rast_is_c_null_value(&tmp_buf[col])) {
-		*has_null = 1;
-		BM_set(nullmap, col, row, 1);
-	    }
-	    else {
-		val = *ti;
-		if (abs(val) > max_short) {
-		    overflow = 1;
-		    /* assign floor/ceiling value?
-		     */
-		    *ts = (short)(max_short * val / abs(val));
-		}
-		else {
-		    *ts = (short)val;
-		}
-	    }
-
-	    ti++;
-	    ts++;
-	}
-    }
-    G_percent(1, 1, 1);
-    
-    Rast_close(cellfile);
-
-    G_free(tmp_buf);
-
-    return (overflow ? -2 : 1);
-}
-
-/*!
-   \brief Load raster map as integer map
-
-   Calling function must have already allocated space in buff for
-   wind->rows * wind->cols unsigned chars.  
-
-   This routine simply loads the map into a 2d array by repetitve calls
-   to get_map_row.
-
-   Since signs of chars can be tricky, we only load positive chars
-   between 0-255.
-
-   \todo fn body Gs_loadmap_as_float()
-
-   \param wind current window
-   \param map_name raster map name
-   \param[out] buff data buffer
-   \param[out] nullmap null map buffer
-   \param[out] has_null indicates if raster map contains null-data
-
-   \return 1 on success
-   \return -1 on failure
-   \return -2 if read ok, but 1 or more values
-   were too large (small) to fit into an unsigned char.
-   (in which case the max (min) char is used)
- */
-int Gs_loadmap_as_char(struct Cell_head *wind, const char *map_name,
-		       unsigned char *buff, struct BM *nullmap, int *has_null)
-{
-    FILEDESC cellfile;
-    const char *map_set;
-    int *ti, *tmp_buf;
-    int offset, row, col, val, max_char, overflow, charsize, bitplace;
-    unsigned char *tc;
-
-    G_debug(3, "Gs_loadmap_as_char");
-
-    overflow = 0;
-    charsize = 8 * sizeof(unsigned char);
-
-    /* 0 bits for sign! */
-    max_char = 1;
-
-    for (bitplace = 0; bitplace < charsize; ++bitplace) {
-	max_char *= 2;
-    }
-
-    max_char -= 1;
-
-    map_set = G_find_raster2(map_name, "");
-    if (!map_set) {
-	G_warning(_("Raster map <%s> not found"), map_name);
-	return -1;
-    }
-    *has_null = 0;
-
-    cellfile = Rast_open_old(map_name, map_set);
-
-    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
-    if (!tmp_buf) {
-	return -1;
-    }
-
-    G_message(_("Loading raster map <%s>..."),
-	      G_fully_qualified_name(map_name, map_set));
-
-    for (row = 0; row < wind->rows; row++) {
-	offset = row * wind->cols;
-	Rast_get_c_row(cellfile, tmp_buf, row);
-	tc = (unsigned char *)&(buff[offset]);
-	ti = tmp_buf;
-
-	G_percent(row, wind->rows, 2);
-
-	for (col = 0; col < wind->cols; col++) {
-	    if (Rast_is_c_null_value(&tmp_buf[col])) {
-		*has_null = 1;
-		BM_set(nullmap, col, row, 1);
-	    }
-	    else {
-		val = *ti;
-		if (val > max_char) {
-		    overflow = 1;
-		    *tc = (unsigned char)max_char;
-		}
-		else if (val < 0) {
-		    overflow = 1;
-		    *tc = 0;
-		}
-		else {
-		    *tc = (unsigned char)val;
-		}
-	    }
-
-	    ti++;
-	    tc++;
-	}
-    }
-    G_percent(1, 1, 1);
-    
-    Rast_close(cellfile);
-
-    G_free(tmp_buf);
-
-    return (overflow ? -2 : 1);
-}
-
-/*!
-   \brief Load raster map as integer map
-
-   Calling function must have already allocated space in buff for
-   struct BM of wind->rows & wind->cols.
-
-   This routine simply loads the map into the bitmap by repetitve calls
-   to get_map_row.  Any value other than 0 in the map will set the bitmap.
-   (may want to change later to allow specific value to set)
-
-   Changed to use null.
-
-   \param wind current window
-   \param map_name raster map name
-   \param[out] buff data buffer
-
-   \returns 1 on success
-   \return -1 on failure
- */
-int Gs_loadmap_as_bitmap(struct Cell_head *wind, const char *map_name,
-			 struct BM *buff)
-{
-    FILEDESC cellfile;
-    const char *map_set;
-    int *tmp_buf;
-    int row, col;
-
-    G_debug(3, "Gs_loadmap_as_bitmap");
-
-    map_set = G_find_raster2(map_name, "");
-    if (!map_set) {
-	G_warning(_("Raster map <%s> not found"), map_name);
-	return -1;
-    }
-
-    cellfile = Rast_open_old(map_name, map_set);
-
-    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
-    if (!tmp_buf) {
-	return -1;
-    }
-
-    G_message(_("Loading raster map <%s>..."),
-	      G_fully_qualified_name(map_name, map_set));
-
-    for (row = 0; row < wind->rows; row++) {
-	Rast_get_c_row(cellfile, tmp_buf, row);
-
-	for (col = 0; col < wind->cols; col++) {
-	    if (Rast_is_c_null_value(&tmp_buf[col])) {
-		/* no data */
-		BM_set(buff, col, row, 1);
-	    }
-	    else {
-		BM_set(buff, col, row, 0);
-	    }
-	}
-    }
-
-    Rast_close(cellfile);
-
-    G_free(tmp_buf);
-
-    return (1);
-}
-
-/*!
-   \brief Build color table (256)
-
-   Calling function must have already allocated space in buff for range of
-   data (256 for now) - simply calls get_color for each cat in color range
-
-   \param filename raster map name
-   \param[out] buff data buffer
-
-   \return 1 on success
-   \return 0 on failure
- */
-int Gs_build_256lookup(const char *filename, int *buff)
-{
-    const char *mapset;
-    struct Colors colrules;
-    CELL min, max, cats[256];
-    int i;
-    unsigned char r[256], g[256], b[256], set[256];
-
-    G_debug(3, "building color table");
-
-    mapset = G_find_raster2(filename, "");
-    if (!mapset) {
-	G_warning(_("Raster map <%s> not found"), filename);
-	return 0;
-    }
-
-    Rast_read_colors(filename, mapset, &colrules);
-    Rast_get_c_color_range(&min, &max, &colrules);
-
-    if (min < 0 || max > 255) {
-	G_warning(_("Color table range doesn't match data (mincol=%d, maxcol=%d"),
-		  min, max);
-
-	min = min < 0 ? 0 : min;
-	max = max > 255 ? 255 : max;
-    }
-
-    G_zero(cats, 256 * sizeof(CELL));
-
-    for (i = min; i <= max; i++) {
-	cats[i] = i;
-    }
-
-    Rast_lookup_c_colors(cats, r, g, b, set, 256, &colrules);
-
-    for (i = 0; i < 256; i++) {
-
-	if (set[i]) {
-	    buff[i] =
-		(r[i] & 0xff) | ((g[i] & 0xff) << 8) | ((b[i] & 0xff) << 16);
-	}
-	else {
-	    buff[i] = NO_DATA_COL;
-	}
-    }
-
-    return (1);
-}
-
-/*!
-   \brief Pack color table
-
-   Passed an array of 32 bit ints that is converted from cell values
-   to packed colors (0xbbggrr) 
-
-   \param filename raster map name
-   \param buff
-   \param rows number of rows
-   \param cols number of cols
- */
-void Gs_pack_colors(const char *filename, int *buff, int rows, int cols)
-{
-    const char *mapset;
-    struct Colors colrules;
-    unsigned char *r, *g, *b, *set;
-    int *cur, i, j;
-
-    mapset = G_find_raster2(filename, "");
-    if (!mapset) {
-	G_warning(_("Raster map <%s> not found"), filename);
-	return;
-    }
-
-    r = (unsigned char *)G_malloc(cols);
-    g = (unsigned char *)G_malloc(cols);
-    b = (unsigned char *)G_malloc(cols);
-    set = (unsigned char *)G_malloc(cols);
-
-    Rast_read_colors(filename, mapset, &colrules);
-
-    cur = buff;
-
-    G_message(_("Translating colors from raster map <%s>..."),
-	      G_fully_qualified_name(filename, mapset));
-
-    for (i = 0; i < rows; i++) {
-	Rast_lookup_c_colors(cur, r, g, b, set, cols, &colrules);
-	G_percent(i, rows, 2);
-
-	for (j = 0; j < cols; j++) {
-	    if (set[j]) {
-		cur[j] =
-		    (r[j] & 0xff) | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) <<
-							    16);
-	    }
-	    else {
-		cur[j] = NO_DATA_COL;
-	    }
-	}
-
-	cur = &(cur[cols]);
-    }
-    G_percent(1, 1, 1);
-    
-    Rast_free_colors(&colrules);
-
-    G_free(r);
-    G_free(g);
-    G_free(b);
-
-    G_free(set);
-
-    return;
-}
-
-/*!
-   \brief Pack color table (floating-point map)
-
-   Passed a array of floats that will be converted from cell values
-   to packed colors (0xbbggrr) and float to int 
-   Floating point data not freed here, use: 
-   gsds_free_data_buff(id, ATTY_FLOAT)
-
-   \param filename raster map name
-   \param fbuf
-   \param ibuf
-   \param rows number of rows
-   \param cols number of cols
- */
-void Gs_pack_colors_float(const char *filename, float *fbuf, int *ibuf,
-			  int rows, int cols)
-{
-    const char *mapset;
-    struct Colors colrules;
-    unsigned char *r, *g, *b, *set;
-    int i, j, *icur;
-    FCELL *fcur;
-
-    mapset = G_find_raster2(filename, "");
-    if (!mapset) {
-	G_warning(_("Raster map <%s> not found"), filename);
-	return;
-    }
-
-    r = (unsigned char *)G_malloc(cols);
-    g = (unsigned char *)G_malloc(cols);
-    b = (unsigned char *)G_malloc(cols);
-    set = (unsigned char *)G_malloc(cols);
-
-    Rast_read_colors(filename, mapset, &colrules);
-
-    fcur = fbuf;
-    icur = ibuf;
-
-    G_message(_("Translating colors from raster map <%s>..."),
-	      G_fully_qualified_name(filename, mapset));
-    
-    for (i = 0; i < rows; i++) {
-	Rast_lookup_f_colors(fcur, r, g, b, set, cols, &colrules);
-	G_percent(i, rows, 2);
-
-	for (j = 0; j < cols; j++) {
-	    if (set[j]) {
-		icur[j] =
-		    (r[j] & 0xff) | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) <<
-							    16);
-	    }
-	    else {
-		icur[j] = NO_DATA_COL;
-	    }
-	}
-
-	icur = &(icur[cols]);
-	fcur = &(fcur[cols]);
-    }
-    G_percent(1, 1, 1);
-    
-    Rast_free_colors(&colrules);
-
-    G_free(r);
-    G_free(g);
-    G_free(b);
-    G_free(set);
-
-    return;
-}
-
-/*!
-   \brief Get categories/labels
-
-   Formats label as in d.what.rast -> (catval) catlabel 
-
-   \param filename raster map name
-   \param drow
-   \param dcol
-   \param catstr category string
-
-   \return 1 on success
-   \return 0 on failure
- */
-int Gs_get_cat_label(const char *filename, int drow, int dcol, char *catstr)
-{
-    struct Categories cats;
-    const char *mapset;
-    CELL *buf;
-    DCELL *dbuf;
-    RASTER_MAP_TYPE map_type;
-    int fd = -1;
-
-    if ((mapset = G_find_raster2(filename, "")) == NULL) {
-	G_warning(_("Raster map <%s> not found"), filename);
-	return 0;
-    }
-
-    if (-1 != Rast_read_cats(filename, mapset, &cats)) {
-	fd = Rast_open_old(filename, mapset);
-	map_type = Rast_get_map_type(fd);
-
-	if (map_type == CELL_TYPE) {
-	    buf = Rast_allocate_c_buf();
-
-	    Rast_get_c_row(fd, buf, drow);
-	    if (Rast_is_c_null_value(&buf[dcol])) {
-		sprintf(catstr, "(NULL) %s",
-			Rast_get_c_cat(&buf[dcol], &cats));
-	    }
-	    else {
-		sprintf(catstr, "(%d) %s", buf[dcol],
-			Rast_get_c_cat(&buf[dcol], &cats));
-	    }
-
-	    G_free(buf);
-	}
-
-	else {
-	    /* fp map */
-	    dbuf = Rast_allocate_d_buf();
-
-	    Rast_get_d_row(fd, dbuf, drow);
-	    if (Rast_is_d_null_value(&dbuf[dcol])) {
-		sprintf(catstr, "(NULL) %s",
-			Rast_get_d_cat(&dbuf[dcol], &cats));
-	    }
-	    else {
-		sprintf(catstr, "(%g) %s", dbuf[dcol],
-			Rast_get_d_cat(&dbuf[dcol], &cats));
-	    }
-
-	    G_free(dbuf);
-	}
-    }
-    else {
-	strcpy(catstr, "no category label");
-	return 0;
-    }
-
-    /* TODO: may want to keep these around for multiple queries */
-    Rast_free_cats(&cats);
-
-    if (fd >= 0)
-	Rast_close(fd);
-
-    return (1);
-}
-
-/*!
-   \brief Save 3dview
-
-   \param vname view name
-   \param gv pointer to geoview struct
-   \param gd pointer to geodisplay struct
-   \param w current window
-   \param defsurf default geosurf struct
-
-   \return -1 on error
-   \return ?
- */
-int Gs_save_3dview(const char *vname, geoview * gv, geodisplay * gd,
-		   struct Cell_head *w, geosurf * defsurf)
-{
-    const char *mapset;
-    struct G_3dview v;
-    float zmax, zmin;
-
-    GS_get_zrange(&zmin, &zmax, 0);
-
-    G_get_3dview_defaults(&v, w);
-    mapset = G_mapset();
-
-    if (mapset != NULL) {
-	if (defsurf) {
-	    if (defsurf->draw_mode & DM_WIRE_POLY) {
-		v.display_type = 3;
-	    }
-	    else if (defsurf->draw_mode & DM_WIRE ||
-		     defsurf->draw_mode & DM_COL_WIRE) {
-		v.display_type = 1;
-	    }
-	    else if (defsurf->draw_mode & DM_POLY) {
-		v.display_type = 2;
-	    }
-
-	    v.mesh_freq = defsurf->x_modw;	/* mesh resolution */
-	    v.poly_freq = defsurf->x_mod;	/* poly resolution */
-	    v.dozero = !(defsurf->nz_topo);
-	    v.colorgrid = (defsurf->draw_mode & DM_COL_WIRE) ? 1 : 0;
-	    v.shading = (defsurf->draw_mode & DM_GOURAUD) ? 1 : 0;
-	}
-
-	if (gv->infocus) {
-	    GS_v3eq(v.from_to[TO], gv->real_to);
-	    v.from_to[TO][Z] -= zmin;
-	    GS_v3mult(v.from_to[TO], gv->scale);
-	    v.from_to[TO][Z] *= gv->vert_exag;
-	}
-	else {
-	    GS_v3eq(v.from_to[TO], gv->from_to[TO]);
-	}
-
-	gsd_model2real(v.from_to[TO]);
-
-	GS_v3eq(v.from_to[FROM], gv->from_to[FROM]);
-	gsd_model2real(v.from_to[FROM]);
-
-	v.exag = gv->vert_exag;
-	v.fov = gv->fov / 10.;
-	v.twist = gv->twist;
-	v.fringe = 0;		/* not implemented here */
-
-	v.lightson = 1;		/* always true, curently */
-
-	if (gv->lights[0].position[W] == 1) {
-	    /* local */
-	    v.lightpos[X] = gv->lights[0].position[X];
-	    v.lightpos[Y] = gv->lights[0].position[Y];
-	    v.lightpos[Z] = gv->lights[0].position[Z];
-	    gsd_model2real(v.lightpos);
-	    v.lightpos[W] = 1.0;	/* local */
-	}
-	else {
-	    v.lightpos[X] = gv->lights[0].position[X];
-	    v.lightpos[Y] = gv->lights[0].position[Y];
-	    v.lightpos[Z] = gv->lights[0].position[Z];
-	    v.lightpos[W] = 0.0;	/* inf */
-	}
-
-	v.lightcol[0] = gv->lights[0].color[0];
-	v.lightcol[1] = gv->lights[0].color[1];
-	v.lightcol[2] = gv->lights[0].color[2];
-
-	v.ambient = (gv->lights[0].ambient[0] + gv->lights[0].ambient[1] +
-		     gv->lights[0].ambient[2]) / 3.;
-	v.shine = gv->lights[0].shine;
-
-	v.surfonly = 0;		/* N/A - now uses constant color */
-	strcpy((v.pgm_id), "Nvision-ALPHA!");
-
-	return (G_put_3dview(vname, mapset, &v, w));
-    }
-    else {
-	return (-1);
-    }
-}
-
-/*!
-   \brief Load 3dview
-
-   \param vname view name
-   \param gv pointer to geoview struct
-   \param gd pointer to geodisplay struct
-   \param w current window
-   \param defsurf default geosurf struct
-
-   \return 1
- */
-int Gs_load_3dview(const char *vname, geoview * gv, geodisplay * gd,
-		   struct Cell_head *w, geosurf * defsurf)
-{
-    const char *mapset;
-    struct G_3dview v;
-    int ret = -1;
-    float pt[3];
-
-    mapset = G_find_file2("3d.view", vname, "");
-
-    if (mapset != NULL) {
-	ret = G_get_3dview(vname, mapset, &v);
-    }
-
-    if (ret >= 0) {
-	if (strcmp((v.pgm_id), "Nvision-ALPHA!")) {
-	    G_warning(_("View not saved by this program,"
-			"there may be some inconsistancies"));
-	}
-
-	/* set poly and mesh resolutions */
-	v.mesh_freq = (int)(v.mesh_freq * v.vwin.ns_res / w->ns_res);
-	v.poly_freq = (int)(v.poly_freq * v.vwin.ns_res / w->ns_res);
-
-	/* Set To and FROM positions */
-	/* TO */
-	pt[0] = (v.from_to[TO][X] - w->west) - w->ew_res / 2.;
-	pt[1] = (v.from_to[TO][Y] - w->south) - w->ns_res / 2.;
-	pt[2] = v.from_to[TO][Z];
-	GS_set_focus(pt);
-
-	/* FROM */
-	pt[0] = (float)v.from_to[FROM][X];
-	pt[1] = (float)v.from_to[FROM][Y];
-	pt[2] = (float)v.from_to[FROM][Z];
-	GS_moveto_real(pt);
-
-	if (defsurf) {
-	    int dmode = 0;
-
-	    GS_setall_drawres(v.poly_freq, v.poly_freq,
-			      v.mesh_freq, v.mesh_freq);
-
-	    while (v.display_type >= 10) {
-		/* globe stuff not used */
-		v.display_type -= 10;
-	    }
-
-	    /* set drawing modes */
-	    if (v.colorgrid) {
-		dmode |= DM_COL_WIRE;
-	    }
-
-	    if (v.shading) {
-		dmode |= DM_GOURAUD;
-	    }
-
-	    switch (v.display_type) {
-	    case 1:
-		dmode |= DM_WIRE;
-
-		break;
-	    case 2:
-		dmode |= DM_POLY;
-
-		break;
-	    case 3:
-		dmode |= DM_WIRE_POLY;
-
-		break;
-	    }
-	    GS_setall_drawmode(dmode);
-
-	    /* should also set nozeros here */
-	}
-
-	/* set exaggeration */
-	if (v.exag)
-	    GS_set_global_exag(v.exag);
-
-	/* Set FOV */
-	if (v.fov) {
-	    GS_set_fov((int)
-		       (v.fov > 0 ? v.fov * 10. + 0.5 : v.fov * 10. - 0.5));
-	}
-	else {
-	    /* TODO: do ortho */
-	}
-
-	/* Set twist */
-	if (v.twist)
-	    GS_set_twist((int)(v.twist > 0 ? v.twist + 0.5 : v.twist - 0.5));
-
-
-	/* TODO:  OK to here - need to unravel/reverse lights stuff*** */
-
-	if (v.lightson) {
-	    /* Lights are on */
-
-	    /* Light Position */
-	    gv->lights[0].position[X] = v.lightpos[X];
-	    gv->lights[0].position[Y] = v.lightpos[Y];
-	    gv->lights[0].position[Z] = v.lightpos[Z];
-
-	    /* Light Color */
-	    gv->lights[0].color[0] = v.lightcol[0];
-	    gv->lights[0].color[1] = v.lightcol[1];
-	    gv->lights[0].color[2] = v.lightcol[2];
-
-	    /* Light Shininess */
-	    gv->lights[0].shine = v.shine;
-
-	    /* Light Ambient */
-	    gv->lights[0].ambient[0] = gv->lights[0].ambient[1] =
-		gv->lights[0].ambient[2] = v.ambient * 3.;
-
-
-	}			/* Done with lights */
-
-
-	GS_alldraw_wire();
-
-    }				/* Done with file */
-    return (1);
-
-}
-
-/*!
-   \brief Update no_zero ranges for attribute (actually no_null now)
-
-   \param gs pointer to geosurf struct
-   \param desc attribute id (descriptor)
-
-   \return -1 on error
-   \return 1 on success
- */
-int Gs_update_attrange(geosurf * gs, int desc)
-{
-    long size;
-    float min, max;
-    typbuff *tb;
-    struct BM *nm;
-    int found;
-
-    gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].range_nz =
-	0.0;
-
-    if (CONST_ATT == gs_get_att_src(gs, desc)) {
-	gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].constant;
-	min = max = gs->att[desc].constant;
-	gs->att[desc].range_nz = 0.0;
-    }
-    else if (CF_COLOR_PACKED & gsds_get_changed(gs->att[desc].hdata)) {
-	gs->att[desc].max_nz = 0xFFFFFF;
-	gs->att[desc].min_nz = 0x010101;
-	gs->att[desc].range_nz = 0xFFFFFF;
-    }
-    else {
-	if (NULL == (tb = gsds_get_typbuff(gs->att[desc].hdata, 0))) {
-	    return (-1);
-	}
-
-	nm = tb->nm;
-
-	if (tb->ib) {
-	    int *p;
-
-	    size = gs->rows * gs->cols;
-	    p = tb->ib;
-	    INIT_MINMAX(p, nm, size, min, max, found);
-
-	    if (!found) {
-		/* all nulls! */
-		return (-1);
-	    }
-
-	    size = gs->rows * gs->cols;
-	    p = tb->ib;
-	    SET_MINMAX(p, nm, size, min, max);
-	}
-	else if (tb->sb) {
-	    short *p;
-
-	    size = gs->rows * gs->cols;
-	    p = tb->sb;
-	    INIT_MINMAX(p, nm, size, min, max, found);
-
-	    if (!found) {
-		/* all nulls! */
-		return (-1);
-	    }
-
-	    size = gs->rows * gs->cols;
-	    p = tb->sb;
-	    SET_MINMAX(p, nm, size, min, max);
-	}
-	else if (tb->cb) {
-	    char *p;
-
-	    size = gs->rows * gs->cols;
-	    p = (char *)tb->cb;
-	    INIT_MINMAX(p, nm, size, min, max, found);
-
-	    if (!found) {
-		/* all nulls! */
-		return (-1);
-	    }
-
-	    size = gs->rows * gs->cols;
-	    p = (char *)tb->cb;
-	    SET_MINMAX(p, nm, size, min, max);
-	}
-	else if (tb->fb) {
-	    float *p;
-
-	    size = gs->rows * gs->cols;
-	    p = tb->fb;
-	    INIT_MINMAX(p, nm, size, min, max, found);
-
-	    if (!found) {
-		/* all nulls! */
-		return (-1);
-	    }
-
-	    size = gs->rows * gs->cols;
-	    p = tb->fb;
-	    SET_MINMAX(p, nm, size, min, max);
-	}
-
-	gs->att[desc].max_nz = max;
-	gs->att[desc].min_nz = min;
-	gs->att[desc].range_nz = gs->att[desc].max_nz - gs->att[desc].min_nz;
-    }
-
-    if (ATT_TOPO == desc) {
-	gs->zmin = min;
-	gs->zmax = max;
-	gs->zrange = gs->zmax - gs->zmin;
-	gs->zminmasked = gs->zmin;
-	gs->zmax_nz = gs->zmax;
-	gs->zmin_nz = gs->zmin;
-	gs->zrange_nz = gs->zmax_nz - gs->zmin_nz;
-    }
-
-    G_debug(3, "Gs_update_attrange(): min=%f max=%f", gs->zmin, gs->zmax);
-
-    return (1);
-}

Deleted: grass/trunk/lib/ogsf/Gv3.c
===================================================================
--- grass/trunk/lib/ogsf/Gv3.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/Gv3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,419 +0,0 @@
-/*!
-   \file lib/ogsf/Gv3.c
-
-   \brief OGSF library - loading vector sets (lower level functions)
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008, 2011 by the GRASS Development Team
-
-   This program is free software under the GNU General Public License
-   (>=v2). Read the file COPYING that comes with GRASS for details.
-
-   \author Bill Brown USACERL (December 1993)
-   \author Updated by Martin Landa <landa.martin gmail.com>
-   (doxygenized in May 2008, thematic mapping in August 2011)
- */
-
-#include <stdlib.h>
-
-#include <grass/gis.h>
-#include <grass/colors.h>
-#include <grass/raster.h>
-#include <grass/vector.h>
-#include <grass/dbmi.h>
-#include <grass/glocale.h>
-#include <grass/ogsf.h>
-
-/*
-   #define TRAK_MEM
-*/
-
-#ifdef TRAK_MEM
-static int Tot_mem = 0;
-#endif
-
-/*!
-   \brief Load vector map to memory
-
-   The other alternative may be to load to a tmp file
-
-   \param grassname vector map name
-   \param[out] number of loaded features
-
-   \return pointer to geoline struct
-   \return NULL on failure
- */
-geoline *Gv_load_vect(const char *grassname, int *nlines)
-{
-    struct Map_info map;
-    struct line_pnts *points;
-    struct line_cats *Cats = NULL;
-    geoline *top, *gln, *prev;
-    int np, i, n, nareas, nl = 0, area, type, is3d;
-    struct Cell_head wind;
-    float vect[2][3];
-    const char *mapset;
-
-    mapset = G_find_vector2(grassname, "");
-    if (!mapset) {
-	G_warning(_("Vector map <%s> not found"), grassname);
-	return NULL;
-    }
-
-    Vect_set_open_level(2);
-    if (Vect_open_old(&map, grassname, "") == -1) {
-	G_warning(_("Unable to open vector map <%s>"),
-		  G_fully_qualified_name(grassname, mapset));
-	return NULL;
-    }
-
-    top = gln = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
-    if (!top) {
-	return NULL;
-    }
-
-    prev = top;
-
-#ifdef TRAK_MEM
-    Tot_mem += sizeof(geoline);
-#endif
-
-    points = Vect_new_line_struct();
-    Cats = Vect_new_cats_struct();
-
-    G_get_set_window(&wind);
-    Vect_set_constraint_region(&map, wind.north, wind.south, wind.east,
-			       wind.west, PORT_DOUBLE_MAX, -PORT_DOUBLE_MAX);
-
-    is3d = Vect_is_3d(&map);
-
-    /* Read areas */
-    n = Vect_get_num_areas(&map);
-    nareas = 0;
-    G_debug(3, "Reading vector areas (nareas = %d)", n);
-    for (area = 1; area <= n; area++) {
-	G_debug(3, " area %d", area);
-	Vect_get_area_points(&map, area, points);
-	if (points->n_points < 3)
-	    continue;
-
-	/* initialize style */
-	gln->highlighted = 0;
-
-	gln->type = OGSF_POLYGON;
-	gln->npts = np = points->n_points;
-	G_debug(3, "  np = %d", np);
-
-	if (is3d) {
-	    gln->dims = 3;
-	    gln->p3 = (Point3 *) G_calloc(np, sizeof(Point3));	/* G_fatal_error */
-	    if (!gln->p3) {
-		return (NULL);
-	    }
-#ifdef TRAK_MEM
-	    Tot_mem += (np * sizeof(Point3));
-#endif
-	}
-	else {
-	    gln->dims = 2;
-	    gln->p2 = (Point2 *) G_calloc(np, sizeof(Point2));	/* G_fatal_error */
-	    if (!gln->p2) {
-		return (NULL);
-	    }
-#ifdef TRAK_MEM
-	    Tot_mem += (np * sizeof(Point2));
-#endif
-	}
-
-	for (i = 0; i < np; i++) {
-	    if (is3d) {
-		gln->p3[i][X] = points->x[i];
-		gln->p3[i][Y] = points->y[i];
-		gln->p3[i][Z] = points->z[i];
-	    }
-	    else {
-		gln->p2[i][X] = points->x[i];
-		gln->p2[i][Y] = points->y[i];
-	    }
-	}
-	/* Calc normal (should be average) */
-	if (is3d) {
-	    vect[0][X] = (float)(gln->p3[0][X] - gln->p3[1][X]);
-	    vect[0][Y] = (float)(gln->p3[0][Y] - gln->p3[1][Y]);
-	    vect[0][Z] = (float)(gln->p3[0][Z] - gln->p3[1][Z]);
-	    vect[1][X] = (float)(gln->p3[2][X] - gln->p3[1][X]);
-	    vect[1][Y] = (float)(gln->p3[2][Y] - gln->p3[1][Y]);
-	    vect[1][Z] = (float)(gln->p3[2][Z] - gln->p3[1][Z]);
-	    GS_v3cross(vect[1], vect[0], gln->norm);
-
-	}
-
-	gln->cats = NULL;
-	gln->next = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
-	if (!gln->next) {
-	    return (NULL);
-	}
-
-#ifdef TRAK_MEM
-	Tot_mem += sizeof(geoline);
-#endif
-
-	prev = gln;
-	gln = gln->next;
-	nareas++;
-    }
-    G_debug(3, "%d areas loaded", nareas);
-
-    /* Read all lines */
-    G_debug(3, "Reading vector lines ...");
-    while (-1 < (type = Vect_read_next_line(&map, points, Cats))) {
-	G_debug(3, "line type = %d", type);
-	if (type & (GV_LINES | GV_FACE)) {
-	    if (type & (GV_LINES)) {
-		gln->type = OGSF_LINE;
-	    }
-	    else {
-		gln->type = OGSF_POLYGON;
-		/* Vect_append_point ( points, points->x[0], points->y[0], points->z[0] ); */
-	    }
-
-	    /* initialize style */
-	    gln->highlighted = 0;
-
-	    gln->npts = np = points->n_points;
-	    G_debug(3, "  np = %d", np);
-
-	    if (is3d) {
-		gln->dims = 3;
-		gln->p3 = (Point3 *) G_calloc(np, sizeof(Point3));	/* G_fatal_error */
-		if (!gln->p3) {
-		    return (NULL);
-		}
-#ifdef TRAK_MEM
-		Tot_mem += (np * sizeof(Point3));
-#endif
-	    }
-	    else {
-		gln->dims = 2;
-		gln->p2 = (Point2 *) G_calloc(np, sizeof(Point2));	/* G_fatal_error */
-		if (!gln->p2) {
-		    return (NULL);
-		}
-#ifdef TRAK_MEM
-		Tot_mem += (np * sizeof(Point2));
-#endif
-	    }
-
-	    for (i = 0; i < np; i++) {
-		if (is3d) {
-		    gln->p3[i][X] = points->x[i];
-		    gln->p3[i][Y] = points->y[i];
-		    gln->p3[i][Z] = points->z[i];
-		}
-		else {
-		    gln->p2[i][X] = points->x[i];
-		    gln->p2[i][Y] = points->y[i];
-		}
-	    }
-	    /* Calc normal (should be average) */
-	    if (is3d && gln->type == OGSF_POLYGON) {
-		vect[0][X] = (float)(gln->p3[0][X] - gln->p3[1][X]);
-		vect[0][Y] = (float)(gln->p3[0][Y] - gln->p3[1][Y]);
-		vect[0][Z] = (float)(gln->p3[0][Z] - gln->p3[1][Z]);
-		vect[1][X] = (float)(gln->p3[2][X] - gln->p3[1][X]);
-		vect[1][Y] = (float)(gln->p3[2][Y] - gln->p3[1][Y]);
-		vect[1][Z] = (float)(gln->p3[2][Z] - gln->p3[1][Z]);
-		GS_v3cross(vect[1], vect[0], gln->norm);
-		G_debug(3, "norm %f %f %f", gln->norm[0], gln->norm[1],
-			gln->norm[2]);
-	    }
-
-	    /* Store category info for thematic display */
-	    if (Cats->n_cats > 0) {
-		gln->cats = Cats;
-		Cats = Vect_new_cats_struct();
-	    }
-	    else {
-		gln->cats = NULL;
-		Vect_reset_cats(Cats);
-	    }
-
-	    gln->next = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
-	    if (!gln->next) {
-		return (NULL);
-	    }
-#ifdef TRAK_MEM
-	    Tot_mem += sizeof(geoline);
-#endif
-
-	    prev = gln;
-	    gln = gln->next;
-	    nl++;
-	}
-    }
-    G_debug(3, "%d lines loaded", nl);
-
-    nl += nareas;
-
-    prev->next = NULL;
-    G_free(gln);
-
-#ifdef TRAK_MEM
-    Tot_mem -= sizeof(geoline);
-#endif
-
-    Vect_close(&map);
-
-    if (!nl) {
-	G_warning(_("No features from vector map <%s> fall within current region"),
-		  G_fully_qualified_name(grassname, mapset));
-	return (NULL);
-    }
-    else {
-	G_message(_("Vector map <%s> loaded (%d features)"),
-		  G_fully_qualified_name(grassname, mapset), nl);
-    }
-
-    *nlines = nl;
-
-#ifdef TRAK_MEM
-    G_debug(3, "Total vect memory = %d Kbytes", Tot_mem / 1000);
-#endif
-
-    return (top);
-}
-
-/*! 
-   \brief Tracking memory 
-
-   \param minus mimus number 
- */
-void sub_Vectmem(int minus)
-{
-    G_debug(5, "sub_Vectmem(): minus=%d", minus);
-#ifdef TRAK_MEM
-    {
-	Tot_mem -= minus;
-    }
-#endif
-
-    return;
-}
-
-/*!
-  \brief Load styles for geolines based on thematic mapping
-
-  \param gv pointer to geovect structure
-  \param colors pointer to Colors structure or NULL
-
-  \return number of features defined by thematic mapping
-  \return -1 on error
-*/
-int Gv_load_vect_thematic(geovect *gv, struct Colors *colors)
-{
-    geoline *gvt;
-
-    struct Map_info Map;
-    struct field_info *Fi;
-    
-    int nvals, cat, nlines, nskipped;
-    int red, blu, grn;
-    const char *str;
-    const char *mapset;
-
-    dbDriver *driver;
-    dbValue value;
-    
-    if(!gv || !gv->tstyle || !gv->filename)
-	return -1;
-
-    mapset = G_find_vector2(gv->filename, "");
-    if (!mapset) {
-	G_fatal_error(_("Vector map <%s> not found"), gv->filename);
-    }
-    
-    Vect_set_open_level(1);
-    if (Vect_open_old(&Map, gv->filename, "") == -1) {
-	G_fatal_error(_("Unable to open vector map <%s>"),
-		      G_fully_qualified_name(gv->filename, mapset));
-    }
-    
-    Fi = Vect_get_field(&Map, gv->tstyle->layer);
-    if (!Fi) {
-	G_warning(_("Database connection not defined for layer %d"),
-		  gv->tstyle->layer);
-    }
-    else {
-      driver = db_start_driver_open_database(Fi->driver, Fi->database);
-      if (!driver)
-	  G_fatal_error(_("Unable to open database <%s> by driver <%s>"),
-			Fi->database, Fi->driver);
-    }
-    G_message(_("Loading thematic vector layer <%s>..."),
-	      G_fully_qualified_name(gv->filename, mapset));
-    nlines = nskipped = 0;
-    for(gvt = gv->lines; gvt; gvt = gvt->next) {
-	gvt->style = (gvstyle *) G_malloc(sizeof(gvstyle));
-	G_zero(gvt->style, sizeof(gvstyle));
-	
-	/* use default style */
-	gvt->style->color  = gv->style->color;
-	gvt->style->symbol = gv->style->symbol;
-	gvt->style->size   = gv->style->size;
-	gvt->style->width  = gv->style->width;
-
-	cat = -1;
-	if (gvt->cats)
-	    Vect_cat_get(gvt->cats, gv->tstyle->layer, &cat);
-	if (cat < 0) {
-	    nskipped++;
-	    continue;
-	}
-	
-	/* color */
-	if (colors) {
-	    if (!Rast_get_c_color((const CELL *) &cat, &red, &grn, &blu, colors)) {
-		G_warning(_("No color rule defined for category %d"), cat);
-		gvt->style->color = gv->style->color;
-	    }
-	    gvt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
-		((int)((blu) << 16) & BLU_MASK);
-	}
-	
-	if (gv->tstyle->color_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gv->tstyle->color_column, &value);
-	    if (nvals < 1)
-		continue;
-	    str = db_get_value_string(&value);
-	    if (!str)
-		continue;
-	    if (G_str_to_color(str, &red, &grn, &blu) != 1) {
-		G_warning(_("Invalid color definition (%s)"),
-			  str);
-		gvt->style->color = gv->style->color;
-	    }
-	    else {
-		gvt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
-		    ((int)((blu) << 16) & BLU_MASK);
-	    }
-	}
-	
-	/* width */
-	if (gv->tstyle->width_column) {
-	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gv->tstyle->width_column, &value);
-	    if (nvals < 1)
-		continue;
-	    gvt->style->width = db_get_value_int(&value);
-	}
-
-	nlines++;
-    }
-
-    if (nskipped > 0)
-	G_warning(_("%d features without category. "
-		    "Unable to determine color rules for features without category."),
-		  nskipped);
-    
-    return nlines;
-}

Deleted: grass/trunk/lib/ogsf/Gvl3.c
===================================================================
--- grass/trunk/lib/ogsf/Gvl3.c	2014-10-28 13:23:47 UTC (rev 62434)
+++ grass/trunk/lib/ogsf/Gvl3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -1,88 +0,0 @@
-/*!
-   \file Gvl3.c
-
-   \brief OGSF library - loading volumes (lower level functions)
-
-   GRASS OpenGL gsurf OGSF Library 
-
-   (C) 1999-2008 by the GRASS Development Team
-
-   This program is free software under the 
-   GNU General Public License (>=v2). 
-   Read the file COPYING that comes with GRASS
-   for details.
-
-   \author Tomas Paudits (December 2003)
-   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
- */
-
-#include <grass/gis.h>
-#include <grass/raster.h>
-#include <grass/raster3d.h>
-#include <grass/ogsf.h>
-#include <grass/glocale.h>
-
-/*!
-   \brief Load color table
-
-   \param[out] color_data color data buffer
-   \param name 3D raster map name
-
-   \return -1 on failure
-   \return 1 on success
- */
-int Gvl_load_colors_data(void **color_data, const char *name)
-{
-    const char *mapset;
-    struct Colors *colors;
-
-    if (NULL == (mapset = G_find_raster3d(name, ""))) {
-	G_warning(_("3D raster map <%s> not found"), name);
-	return (-1);
-    }
-
-    if (NULL == (colors = (struct Colors *)G_malloc(sizeof(struct Colors))))
-	return (-1);
-
-    if (0 > Rast3d_read_colors(name, mapset, colors)) {
-	G_free(colors);
-	return (-1);
-    }
-
-    *color_data = colors;
-
-    return (1);
-}
-
-/*!
-   \brief Unload color table
-
-   \param color_data color data buffer
-
-   \return -1 on failure
-   \return 1 on success
- */
-int Gvl_unload_colors_data(void *color_data)
-{
-    Rast_free_colors(color_data);
-
-    G_free(color_data);
-
-    return (1);
-}
-
-/*!
-   \brief Get color for value
-
-   \param color_data color data value
-   \param value data value
-
-   \return color value
- */
-int Gvl_get_color_for_value(void *color_data, float *value)
-{
-    int r, g, b;
-
-    Rast_get_f_color((FCELL *) value, &r, &g, &b, color_data);
-    return ((r & 0xff) | ((g & 0xff) << 8) | ((b & 0xff) << 16));
-}

Copied: grass/trunk/lib/ogsf/gk2.c (from rev 62429, grass/trunk/lib/ogsf/GK2.c)
===================================================================
--- grass/trunk/lib/ogsf/gk2.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gk2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,625 @@
+/*!
+   \file GK2.c
+
+   \brief OGSF library - setting and manipulating keyframes animation
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown USACERL, GMSL/University of Illinois
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <stdlib.h>
+
+#include <grass/gis.h>
+#include <grass/glocale.h>
+#include <grass/ogsf.h>
+
+static int _add_key(Keylist *, int, float);
+static void _remove_key(Keylist *);
+
+static Keylist *Keys = NULL;
+static Keylist *Keytail = NULL;
+static Viewnode *Views = NULL;
+static float Keystartpos = 0.0;
+static float Keyendpos = 1.0;
+static float Tension = 0.8;
+static int Viewsteps = 0;
+static int Numkeys = 0;
+static int Interpmode = KF_SPLINE;
+static int Fmode = 0;
+
+/* next & prior already initialized to NULL */
+static int _add_key(Keylist * newk, int force_replace, float precis)
+{
+    Keylist *k, *tempk, *prev;
+    int found;
+
+    found = 0;
+    prev = NULL;
+
+    /* if(Viewsteps) precis = 0.5/Viewsteps; */
+    for (k = Keys; k; k = k->next) {
+	if (k->pos >= newk->pos - precis && k->pos <= newk->pos + precis) {
+	    if (force_replace) {
+
+		if (k->prior) {
+		    k->prior->next = newk;
+		    newk->prior = prev;
+		}
+		else {
+		    Keys = newk;
+		}
+
+		newk->next = k->next;
+		newk->prior = k->prior;
+		tempk = k;
+		k = newk;
+		free(tempk);
+	    }
+	    else {
+		free(newk);
+	    }
+
+	    return (-1);
+	}
+    }
+
+    if (Keys) {
+	if (newk->pos < Keys->pos) {
+	    /* new will be first */
+	    newk->next = Keys;
+	    Keys->prior = newk;
+	    Keys = newk;
+	}
+	else {
+	    prev = k = Keys;
+	    while (k && !found) {
+		if (k->pos > newk->pos) {
+		    prev->next = newk;
+		    newk->next = k;
+		    newk->prior = prev;
+		    k->prior = newk;
+		    found = 1;
+		}
+
+		prev = k;
+		k = k->next;
+	    }
+	    if (!found) {
+		Keytail = prev->next = newk;
+		newk->prior = prev;
+	    }
+	}
+    }
+    else {
+	Keys = Keytail = newk;
+    }
+
+    ++Numkeys;
+    return (1);
+}
+
+static void _remove_key(Keylist * k)
+{
+    if (k->prior) {
+	k->prior->next = k->next;
+	if (k->next) {
+	    k->next->prior = k->prior;
+	}
+	else {
+	    Keytail = k->prior;
+	}
+    }
+    else {
+	Keys = k->next;
+	if (k->next) {
+	    k->next->prior = NULL;
+	}
+    }
+    k->next = k->prior = NULL;
+
+    return;
+}
+
+/*!
+   \brief Set interpolation mode 
+
+   \param mode interpolation mode (KF_LINEAR or KF_SPLINE)
+
+   \return 1 on success
+   \return -1 on error (invalid interpolation mode)
+ */
+int GK_set_interpmode(int mode)
+{
+    if (KF_LEGAL_MODE(mode)) {
+	Interpmode = mode;
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set value for tension when interpmode is KF_SPLINE. 
+
+   \param tens value tens should be between 0.0; 1.0.
+ */
+void GK_set_tension(float tens)
+{
+    Tension = tens > 1.0 ? 1.0 : (tens < 0.0 ? 0.0 : tens);
+
+    /* for now */
+    if (Views) {
+	GK_update_frames();
+	GS_set_draw(GSD_BACK);
+	GS_ready_draw();
+	GS_clear(GS_background_color());
+	GS_alldraw_wire();
+
+	gk_draw_path(Views, Viewsteps, Keys);
+
+	GS_done_draw();
+    }
+
+    return;
+}
+
+void GK_showtension_start(void)
+{
+    return;
+}
+
+/*!
+   \brief Show tension stop ?
+
+   Use GK_showtension_start/GK_update_tension/GK_showtension_stop to
+   initialize and stop multi-view display of path when changing
+   tension.
+ */
+void GK_showtension_stop(void)
+{
+    return;
+}
+
+/*!
+   \brief Update tension
+ */
+void GK_update_tension(void)
+{
+    if (Views) {
+	GK_update_frames();
+    }
+
+    return;
+}
+
+/*!
+   \brief Print keyframe info
+
+   \param name filename
+ */
+void GK_print_keys(const char *name)
+{
+    Keylist *k;
+    FILE *fp;
+    int cnt = 1;
+
+    if (NULL == (fp = fopen(name, "w"))) {
+	G_fatal_error(_("Unable to open file <%s> for writing"), name);
+    }
+    /* write a default frame rate of 30 at top of file */
+    fprintf(fp, "30 \n");
+
+    for (k = Keys; k; k = k->next) {
+
+	fprintf(fp,
+		"{%f {{FromX %f} {FromY %f} {FromZ %f} {DirX %f} {DirY %f} {DirZ %f} {FOV %f} {TWIST %f} {cplane-0 {{pos_x 0.000000} {pos_y 0.000000} {pos_z 0.000000} {blend_type OFF} {rot 0.000000} {tilt 0.000000}}}} keyanimtag%d 0} ",
+		k->pos, k->fields[KF_FROMX], k->fields[KF_FROMY],
+		k->fields[KF_FROMZ], k->fields[KF_DIRX], k->fields[KF_DIRY],
+		k->fields[KF_DIRZ], k->fields[KF_FOV] / 10.,
+		k->fields[KF_TWIST], cnt);
+	cnt++;
+    }
+
+    fclose(fp);
+    return;
+
+}
+
+/*!
+   \brief Recalculate path using the current number of frames requested.
+
+   Call after changing number of frames or when
+   Keyframes change.
+ */
+void GK_update_frames(void)
+{
+    Keylist *k;
+    int loop = 0;
+
+    if (Keys) {
+	if (Numkeys > 1) {
+	    k = Keytail;
+	    Keyendpos = k->pos;
+
+	    if (k->fields[KF_FROMX] == Keys->fields[KF_FROMX] &&
+		k->fields[KF_FROMY] == Keys->fields[KF_FROMY] &&
+		k->fields[KF_FROMZ] == Keys->fields[KF_FROMZ]) {
+		loop = 1;
+	    }
+	}
+
+	Keystartpos = Keys->pos;
+    }
+
+    if (Interpmode == KF_LINEAR && Numkeys > 1) {
+	if (Views) {
+	    free(Views);
+	    Views = NULL;
+	}
+
+	Views = gk_make_linear_framesfromkeys(Keys, Numkeys, Viewsteps, loop);
+
+	if (!Views) {
+	    G_warning(_("Check no. of frames requested and keyframes marked"));
+	}
+    }
+    else if (Numkeys > 2) {
+	if (Views) {
+	    free(Views);
+	    Views = NULL;
+	}
+
+	Views = gk_make_framesfromkeys
+	    (Keys, Numkeys, Viewsteps, loop, 1.0 - Tension);
+
+	if (!Views) {
+	    G_warning(_("Check no. of frames requested and keyframes marked"));
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Set the number of frames to be interpolated from keyframes
+
+   \param newsteps number of frames
+ */
+void GK_set_numsteps(int newsteps)
+{
+    Viewsteps = newsteps;
+    GK_update_frames();
+
+    return;
+}
+
+/*!
+   \brief Deletes all keyframes, resets field masks.
+
+   Doesn't change number of frames requested.
+ */
+void GK_clear_keys(void)
+{
+    gk_free_key(Keys);
+    Keys = NULL;
+    Numkeys = 0;
+    free(Views);
+    Views = NULL;
+
+    Keystartpos = 0.0;
+    Keyendpos = 1.0;
+
+    return;
+}
+
+/*!
+   \brief Move keyframe
+
+   Precis works as in other functions - to identify keyframe to move.
+   Only the first keyframe in the precis range will be moved.
+
+   \param oldpos old position
+   \param precis precision value
+   \param newpos new position
+
+   \return number of keys moved (1 or 0)
+ */
+int GK_move_key(float oldpos, float precis, float newpos)
+{
+    Keylist *k;
+
+    for (k = Keys; k; k = k->next) {
+	if (k->pos >= oldpos - precis && k->pos <= oldpos + precis) {
+	    _remove_key(k);
+	    k->pos = newpos;
+	    _add_key(k, 1, precis);
+	    GK_update_frames();
+	    return (1);
+	}
+    }
+
+    return (0);
+}
+
+/*!
+   Delete keyframe
+
+   The values pos and precis are used to determine which keyframes to
+   delete.  Any keyframes with their position within precis of pos will
+   be deleted if justone is zero.  If justone is non-zero, only the first
+   (lowest pos) keyframe in the range will be deleted.
+
+   \param pos position
+   \param precis precision
+   \param justone delete only one keyframe
+
+   \return number of keys deleted.
+ */
+int GK_delete_key(float pos, float precis, int justone)
+{
+    Keylist *k, *next;
+    int cnt;
+
+    for (cnt = 0, k = Keys; k;) {
+	next = k->next;
+
+	if (k->pos >= pos - precis && k->pos <= pos + precis) {
+	    cnt++;
+	    _remove_key(k);
+	    free(k);
+	    if (justone) {
+		break;
+	    }
+	}
+
+	k = next;
+    }
+
+    GK_update_frames();
+    return (cnt);
+}
+
+/*!
+   \brief Add keyframe
+
+   The pos value is the relative position in the animation for this
+   particular keyframe - used to compare relative distance to neighboring
+   keyframes, it can be any floating point value.
+
+   The fmask value can be any of the following or'd together:    
+   - KF_FROMX_MASK    
+   - KF_FROMY_MASK    
+   - KF_FROMZ_MASK    
+   - KF_FROM_MASK (KF_FROMX_MASK | KF_FROMY_MASK | KF_FROMZ_MASK) 
+
+   - KF_DIRX_MASK    
+   - KF_DIRY_MASK    
+   - KF_DIRZ_MASK    
+   - KF_DIR_MASK (KF_DIRX_MASK | KF_DIRY_MASK | KF_DIRZ_MASK) 
+
+   - KF_FOV_MASK    
+   - KF_TWIST_MASK    
+
+   - KF_ALL_MASK (KF_FROM_MASK | KF_DIR_MASK | KF_FOV_MASK | KF_TWIST_MASK) 
+
+   Other fields will be added later.
+
+   The value precis and the boolean force_replace are used to determine
+   if a keyframe should be considered to be at the same position as a
+   pre-existing keyframe. e.g., if anykey.pos - newkey.pos <= precis,
+   GK_add_key() will fail unless force_replace is TRUE.
+
+   \param pos postion
+   \param fmaks
+   \param force_replace
+   \param precis precision value
+
+   \return 1 if key is added
+   \return -1 key not added
+ */
+int GK_add_key(float pos, unsigned long fmask, int force_replace,
+	       float precis)
+{
+    Keylist *newk;
+    float tmp[3];
+
+    if (NULL == (newk = (Keylist *) malloc(sizeof(Keylist)))) {
+	fprintf(stderr, "Out of memory\n");
+	return (-1);
+    }
+
+    /* All fields set, don't use mask until making Views */
+
+    GS_get_from(tmp);
+    newk->fields[KF_FROMX] = tmp[X];
+    newk->fields[KF_FROMY] = tmp[Y];
+    newk->fields[KF_FROMZ] = tmp[Z];
+
+    G_debug(3, "KEY FROM: %f %f %f", tmp[X], tmp[Y], tmp[Z]);
+
+    /* Instead of View Dir try get_focus (view center) */
+    /* View Dir is implied from eye and center position */
+    /*    GS_get_viewdir(tmp); */
+
+    /* ACS 1 line: was      GS_get_focus(tmp);
+       with this kanimator works also for flythrough navigation
+       also changed in gk.c
+     */
+    GS_get_viewdir(tmp);
+    newk->fields[KF_DIRX] = tmp[X];
+    newk->fields[KF_DIRY] = tmp[Y];
+    newk->fields[KF_DIRZ] = tmp[Z];
+
+    newk->fields[KF_FOV] = GS_get_fov();
+    newk->fields[KF_TWIST] = GS_get_twist();
+    newk->pos = pos;
+    newk->fieldmask = fmask;
+    newk->next = NULL;
+    newk->prior = NULL;
+
+    if (0 < _add_key(newk, force_replace, precis)) {
+	GK_update_frames();
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Moves the animation to frame number "step".
+
+   Step should be a value between 1 and the number of frames.  If
+   render is non-zero, calls draw_all.
+
+   \param step step value
+   \param render
+ */
+void GK_do_framestep(int step, int render)
+{
+    if (Views) {
+	if (step > 0 && step <= Viewsteps) {
+	    gk_follow_frames(Views, Viewsteps, Keys, step, 1, render, Fmode);
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw the current path
+
+   \param flag
+ */
+void GK_show_path(int flag)
+{
+    if (flag) {
+	Fmode |= FM_PATH;
+
+	if (Views) {
+	    GS_set_draw(GSD_FRONT);
+	    GS_ready_draw();
+
+	    gk_draw_path(Views, Viewsteps, Keys);
+
+	    GS_done_draw();
+
+	}
+    }
+    else {
+	Fmode &= ~FM_PATH;
+    }
+
+    return;
+}
+
+/*!
+   \brief Show vector sets
+
+   \param flag
+ */
+void GK_show_vect(int flag)
+{
+    if (flag) {
+	Fmode |= FM_VECT;
+	if (Views) {
+
+	    GS_set_draw(GSD_FRONT);
+	    GS_ready_draw();
+
+	    GV_alldraw_vect();
+
+	    GS_done_draw();
+	}
+    }
+    else {
+	Fmode &= ~FM_VECT;
+    }
+
+    return;
+}
+
+/*!
+   \brief Show point sets
+
+   \param flag
+ */
+void GK_show_site(int flag)
+{
+    if (flag) {
+	Fmode |= FM_SITE;
+
+	if (Views) {
+
+	    GS_set_draw(GSD_FRONT);
+	    GS_ready_draw();
+
+	    GP_alldraw_site();
+
+	    GS_done_draw();
+
+	}
+    }
+    else {
+	Fmode &= ~FM_SITE;
+    }
+
+    return;
+}
+
+/*!
+   \brief Show volumes
+
+   \param flag
+ */
+void GK_show_vol(int flag)
+{
+    if (flag) {
+	Fmode |= FM_VOL;
+
+	if (Views) {
+
+	    GS_set_draw(GSD_FRONT);
+	    GS_ready_draw();
+
+	    GVL_alldraw_vol();
+
+	    GS_done_draw();
+
+	}
+    }
+    else {
+	Fmode &= ~FM_VOL;
+    }
+
+    return;
+}
+
+/*!
+   \brief Show list
+
+   \param flag
+ */
+void GK_show_list(int flag)
+{
+    if (flag) {
+	Fmode |= FM_LABEL;
+
+	if (Views) {
+	    GS_draw_all_list();
+	}
+    }
+    else {
+	Fmode &= ~FM_LABEL;
+    }
+
+    return;
+}

Copied: grass/trunk/lib/ogsf/gp2.c (from rev 62429, grass/trunk/lib/ogsf/GP2.c)
===================================================================
--- grass/trunk/lib/ogsf/gp2.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gp2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,710 @@
+/*!
+   \file lib/ogsf/GP2.c
+
+   \brief OGSF library - loading and manipulating point sets (higher level functions)
+
+   (C) 1999-2008, 2011 by the GRASS Development Team
+
+   This program is free software under the GNU General Public License
+   (>=v2). Read the file COPYING that comes with GRASS for details.
+
+   \author Bill Brown USACERL (January 1994)
+   \author Updated by Martin landa <landa.martin gmail.com>
+   (doxygenized in May 2008, thematic mapping in June 2011)
+ */
+
+#include <stdlib.h>
+#include <string.h>
+
+#include <grass/gis.h>
+#include <grass/ogsf.h>
+#include <grass/glocale.h>
+
+#include "gsget.h"
+
+static int Site_ID[MAX_SITES];
+static int Next_site = 0;
+
+/*!
+   \brief Check if point set exists
+
+   \param id point set id
+
+   \return 1 found
+   \return 0 not found
+ */
+int GP_site_exists(int id)
+{
+    int i, found = 0;
+
+    G_debug(4, "GP_site_exists(%d)", id);
+
+    if (NULL == gp_get_site(id)) {
+	return 0;
+    }
+
+    for (i = 0; i < Next_site && !found; i++) {
+	if (Site_ID[i] == id) {
+	    found = 1;
+	}
+    }
+
+    G_debug(3, "GP_site_exists(): found=%d", found);
+
+    return found;
+}
+
+/*!
+   \brief Create new point set
+
+   \return point set id
+   \return -1 on error (number of point sets exceeded)
+ */
+int GP_new_site(void)
+{
+    geosite *np;
+
+    if (Next_site < MAX_SITES) {
+	np = gp_get_new_site();
+	gp_set_defaults(np);
+	Site_ID[Next_site] = np->gsite_id;
+	++Next_site;
+
+	G_debug(3, "GP_new_site() id=%d", np->gsite_id);
+
+	return np->gsite_id;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Get number of loaded point sets
+
+   \return number of point sets
+ */
+int GP_num_sites(void)
+{
+    return gp_num_sites();
+}
+
+/*!
+   \brief Get list of point sets
+
+   Must freed when no longer needed!
+
+   \param numsites number of point sets
+
+   \return pointer to list of points sets
+   \return NULL on error
+ */
+int *GP_get_site_list(int *numsites)
+{
+    int i, *ret;
+
+    *numsites = Next_site;
+
+    if (Next_site) {
+	ret = (int *)G_malloc(Next_site * sizeof(int));	/* G_fatal_error */
+	if (!ret) {
+	    return NULL;
+	}
+
+	for (i = 0; i < Next_site; i++) {
+	    ret[i] = Site_ID[i];
+	}
+
+	return ret;
+    }
+
+    return NULL;
+}
+
+/*!
+   \brief Delete registrated point set
+
+   \param id point set id
+
+   \return 1 on success
+   \return -1 on error (point sets not available)
+ */
+int GP_delete_site(int id)
+{
+    int i, j, found = 0;
+
+    G_debug(4, "GP_delete_site(%d)", id);
+
+    if (GP_site_exists(id)) {
+	gp_delete_site(id);
+
+	for (i = 0; i < Next_site && !found; i++) {
+	    if (Site_ID[i] == id) {
+		found = 1;
+		for (j = i; j < Next_site; j++) {
+		    Site_ID[j] = Site_ID[j + 1];
+		}
+	    }
+	}
+
+	if (found) {
+	    --Next_site;
+	    return 1;
+	}
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Load point set from file
+
+   Check to see if handle already loaded, if so - free before loading
+   new for now, always load to memory.
+
+   \todo load file handle & ready for reading instead of using memory
+
+   \param id point set id
+   \param filename point set filename
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GP_load_site(int id, const char *filename)
+{
+    geosite *gp;
+
+    G_debug(3, "GP_load_site(id=%d, name=%s)", id, filename);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    if (gp->points) {
+	gp_free_sitemem(gp);
+    }
+
+    gp->filename = G_store(filename);
+
+    gp->points = Gp_load_sites(filename, &(gp->n_sites), &(gp->has_z));
+    
+    if (gp->points) {
+	return 1;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Get point set filename
+
+   Note: char array is allocated by G_store()
+
+   \param id point set id
+   \param[out] filename point set filename
+
+   \return -1 on error (point set not found)
+   \return 1 on success
+ */
+int GP_get_sitename(int id, char **filename)
+{
+    geosite *gp;
+
+    G_debug(4, "GP_get_sitename(%d)", id);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    *filename = G_store(gp->filename);
+
+    return 1;
+}
+
+/*!
+   \brief Get point set style
+
+   \param id point set id
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GP_get_style(int id, int *color, int *width, float *size, int *symbol)
+{
+    geosite *gp;
+
+    G_debug(4, "GP_get_style(%d)", id);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    *color = gp->style->color;
+    *width = gp->style->width;
+    *symbol = gp->style->symbol;
+    *size = gp->style->size;
+
+    return 1;
+}
+
+/*!
+   \brief Set point style
+
+   Supported icon symbols (markers):
+    - ST_X
+    - ST_BOX
+    - ST_SPHERE
+    - ST_CUBE
+    - ST_DIAMOND
+    - ST_DEC_TREE
+    - ST_CON_TREE
+    - ST_ASTER
+    - ST_GYRO
+    - ST_HISTOGRAM
+
+   \param id point set id
+   \param color icon color
+   \param width icon line width
+   \param size icon size
+   \param symbol icon symbol
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GP_set_style(int id, int color, int width, float size, int symbol)
+{
+    geosite *gp;
+
+    G_debug(4, "GP_set_style(id=%d, color=%d, width=%d, size=%f, symbol=%d)", id, color, width, size,
+	    symbol);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    gp->style->color = color;
+    gp->style->symbol = symbol;
+    gp->style->size = size;
+    gp->style->width = width;
+
+    return 1;
+}
+
+/*!
+   \brief Set point set style for thematic mapping
+
+   Updates also style for each geopoint.
+   
+   \param id point set id
+   \param layer layer number for thematic mapping (-1 for undefined)
+   \param color icon color column name
+   \param width icon line width column name
+   \param size icon size column name
+   \param symbol icon symbol column name
+   \param colors pointer to Colors structure or NULL
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GP_set_style_thematic(int id, int layer, const char* color, const char* width,
+			  const char* size, const char* symbol, struct Colors *color_rules)
+{
+    geosite *gp;
+    
+    G_debug(4, "GP_set_style_thematic(id=%d, layer=%d, color=%s, width=%s, size=%s, symbol=%s)", id, layer,
+	    color, width, size, symbol);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    if(!gp->tstyle)
+	gp->tstyle = (gvstyle_thematic *)G_malloc(sizeof(gvstyle_thematic));
+    G_zero(gp->tstyle, sizeof(gvstyle_thematic));
+    
+    gp->tstyle->active = 1;
+    gp->tstyle->layer = layer;
+    if (color)
+	gp->tstyle->color_column = G_store(color);
+    if (symbol)
+	gp->tstyle->symbol_column = G_store(symbol);
+    if (size)
+	gp->tstyle->size_column = G_store(size);
+    if (width)
+	gp->tstyle->width_column = G_store(width);
+
+    Gp_load_sites_thematic(gp, color_rules);
+
+    return 1;
+}
+
+/*!
+   \brief Make style for thematic mapping inactive
+   
+   \param id point set id
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GP_unset_style_thematic(int id)
+{
+    geosite *gp;
+
+    G_debug(4, "GP_unset_style_thematic(): id=%d", id);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    if (gp->tstyle) {
+	gp->tstyle->active = 0;
+    }
+
+    return 1;
+}
+
+/*!
+   \brief Set z mode for point set
+
+   \param id point set id
+   \param use_z TRUE to use z-coordinaces when vector map is 3D
+
+   \return 1 on success
+   \return 0 vector map is not 3D
+   \return -1 on error (invalid point set id)
+ */
+/* I don't see who is using it? Why it's required? */
+int GP_set_zmode(int id, int use_z)
+{
+    geosite *gp;
+
+    G_debug(3, "GP_set_zmode(%d,%d)", id, use_z);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    if (use_z) {
+	if (gp->has_z) {
+	    gp->use_z = 1;
+	    return 1;
+	}
+
+	return 0;
+    }
+
+    gp->use_z = 0;
+    return 1;
+}
+
+/*!
+   \brief Get z-mode
+
+   \todo Who's using this?
+   
+   \param id point set id
+   \param[out] use_z non-zero code to use z
+
+   \return -1 on error (invalid point set id)
+   \return 1 on success
+ */
+int GP_get_zmode(int id, int *use_z)
+{
+    geosite *gp;
+
+    G_debug(4, "GP_get_zmode(%d)", id);
+
+    if (NULL == (gp = gp_get_site(id))) {
+	return -1;
+    }
+
+    *use_z = gp->use_z;
+    return 1;
+}
+
+/*!
+   \brief Set transformation params
+
+   \param id point set id
+   \param xtrans,ytrans,ztrans x/y/z values
+ */
+void GP_set_trans(int id, float xtrans, float ytrans, float ztrans)
+{
+    geosite *gp;
+
+    G_debug(3, "GP_set_trans(): id=%d trans=%f,%f,%f",
+	    id, xtrans, ytrans, ztrans);
+
+    gp = gp_get_site(id);
+    if (gp) {
+	gp->x_trans = xtrans;
+	gp->y_trans = ytrans;
+	gp->z_trans = ztrans;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get transformation params
+
+   \param id point set id
+   \param[out] xtrans,ytrans,ztrans x/y/z values
+ */
+void GP_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
+{
+    geosite *gp;
+
+    gp = gp_get_site(id);
+
+    if (gp) {
+	*xtrans = gp->x_trans;
+	*ytrans = gp->y_trans;
+	*ztrans = gp->z_trans;
+    }
+
+    G_debug(3, "GP_get_trans(): id=%d, trans=%f,%f,%f",
+	    id, *xtrans, *ytrans, *ztrans);
+
+    return;
+}
+
+/*!
+   \brief Select surface for given point set
+
+   \param hp point set id
+   \param hs surface id
+
+   \return 1 surface selected
+   \return -1 on error
+ */
+int GP_select_surf(int hp, int hs)
+{
+    geosite *gp;
+
+    G_debug(3, "GP_select_surf(%d,%d)", hp, hs);
+
+    if (GP_surf_is_selected(hp, hs)) {
+	return 1;
+    }
+
+    gp = gp_get_site(hp);
+
+    if (gp && GS_surf_exists(hs)) {
+	gp->drape_surf_id[gp->n_surfs] = hs;
+	gp->n_surfs += 1;
+	return 1;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Unselect surface
+
+   \param hp point set id
+   \param hs surface id
+
+   \return 1 surface unselected
+   \return -1 on error
+ */
+int GP_unselect_surf(int hp, int hs)
+{
+    geosite *gp;
+    int i, j;
+
+    G_debug(3, "GP_unselect_surf(%d,%d)", hp, hs);
+
+    if (!GP_surf_is_selected(hp, hs)) {
+	return 1;
+    }
+
+    gp = gp_get_site(hp);
+
+    if (gp) {
+	for (i = 0; i < gp->n_surfs; i++) {
+	    if (gp->drape_surf_id[i] == hs) {
+		for (j = i; j < gp->n_surfs - 1; j++) {
+		    gp->drape_surf_id[j] = gp->drape_surf_id[j + 1];
+		}
+
+		gp->n_surfs -= 1;
+		return 1;
+	    }
+	}
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Check if surface is selected
+
+   \param hp point set id
+   \param hs surface id
+
+   \return 1 selected
+   \return 0 not selected
+ */
+int GP_surf_is_selected(int hp, int hs)
+{
+    int i;
+    geosite *gp;
+
+    G_debug(3, "GP_surf_is_selected(%d,%d)", hp, hs);
+
+    gp = gp_get_site(hp);
+
+    if (gp) {
+	for (i = 0; i < gp->n_surfs; i++) {
+	    if (hs == gp->drape_surf_id[i]) {
+		return 1;
+	    }
+	}
+    }
+
+    return 0;
+}
+
+/*!
+   \brief Draw point set
+
+   \param id point set id
+ */
+void GP_draw_site(int id)
+{
+    geosurf *gs;
+    geosite *gp;
+    int i;
+    float n, yo, xo, e;
+
+    gp = gp_get_site(id);
+    GS_get_region(&n, &yo, &xo, &e);
+
+    /* kind of sloppy - maybe site files should have an origin, too */
+    if (gp) {
+	if (gp->use_z && gp->has_z) {
+	    gpd_3dsite(gp, xo, yo, 0);
+	}
+	else {
+	    for (i = 0; i < gp->n_surfs; i++) {
+		gs = gs_get_surf(gp->drape_surf_id[i]);
+
+		if (gs) {
+		    gpd_2dsite(gp, gs, 0);
+		    G_debug(5, "Drawing site %d on Surf %d", id,
+			    gp->drape_surf_id[i]);
+		}
+	    }
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all available point sets
+ */
+void GP_alldraw_site(void)
+{
+    int id;
+
+    for (id = 0; id < Next_site; id++) {
+	GP_draw_site(Site_ID[id]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Set client data
+
+   \param id point set id
+   \param clientd client data
+
+   \return 1 on success
+   \return -1 on error (invalid point set id)
+ */
+int GP_Set_ClientData(int id, void *clientd)
+{
+    geosite *gp;
+
+    gp = gp_get_site(id);
+
+    if (gp) {
+	gp->clientdata = clientd;
+	return 1;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Get client data
+
+   \param id point set id
+
+   \return pointer to client data
+   \return NULL on error
+ */
+void *GP_Get_ClientData(int id)
+{
+    geosite *gp;
+
+    gp = gp_get_site(id);
+    if (gp) {
+	return (gp->clientdata);
+    }
+
+    return NULL;
+}
+
+/*!
+  \brief Determine point marker symbol for string
+
+  Supported markers:
+    - ST_X
+    - ST_BOX
+    - ST_SPHERE
+    - ST_CUBE
+    - ST_DIAMOND
+    - ST_DEC_TREE
+    - ST_CON_TREE
+    - ST_ASTER
+    - ST_GYRO
+    - ST_HISTOGRAM
+
+  \param str string buffer
+
+  \return marker code (default: ST_SPHERE)
+*/
+int GP_str_to_marker(const char *str)
+{
+    int marker;
+
+    if (strcmp(str, "x") == 0)
+	marker = ST_X;
+    else if (strcmp(str, "box") == 0)
+	marker = ST_BOX;
+    else if (strcmp(str, "sphere") == 0)
+	marker = ST_SPHERE;
+    else if (strcmp(str, "cube") == 0)
+	marker = ST_CUBE;
+    else if (strcmp(str, "diamond") == 0)
+	marker = ST_DIAMOND;
+    else if (strcmp(str, "dec_tree") == 0)
+	marker = ST_DEC_TREE;
+    else if (strcmp(str, "con_tree") == 0)
+	marker = ST_CON_TREE;
+    else if (strcmp(str, "aster") == 0)
+	marker = ST_ASTER;
+    else if (strcmp(str, "gyro") == 0)
+	marker = ST_GYRO;
+    else if (strcmp(str, "histogram") == 0)
+	marker = ST_HISTOGRAM;
+    else {
+	G_warning(_("Unknown icon marker, using \"sphere\""));
+	marker = ST_SPHERE;
+    }
+
+    return marker;
+}

Copied: grass/trunk/lib/ogsf/gp3.c (from rev 62429, grass/trunk/lib/ogsf/Gp3.c)
===================================================================
--- grass/trunk/lib/ogsf/gp3.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gp3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,292 @@
+/*!
+   \file lib/ogsf/Gp3.c
+
+   \brief OGSF library - loading point sets (lower level functions)
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008, 2011 by the GRASS Development Team
+
+   This program is free software under the GNU General Public License
+   (>=v2).  Read the file COPYING that comes with GRASS for details.
+
+   \author Bill Brown USACERL, GMSL/University of Illinois (January 1994)
+   \author Updated by Martin Landa <landa.martin gmail.com>
+   (doxygenized in May 2008, thematic mapping in June 2011)
+ */
+
+#include <stdlib.h>
+
+#include <grass/gis.h>
+#include <grass/colors.h>
+#include <grass/raster.h>
+#include <grass/vector.h>
+#include <grass/dbmi.h>
+#include <grass/glocale.h>
+#include <grass/ogsf.h>
+
+/*!
+   \brief Load to points to memory
+
+   The other alternative may be to load to a tmp file.
+
+   \param name name of vector map to be loaded
+   \param[out] nsites number of loaded points
+   \param[out] has_z 2D or 3D points data loaded?
+
+   \return pointer to geopoint struct (array)
+   \return NULL on failure
+ */
+geopoint *Gp_load_sites(const char *name, int *nsites, int *has_z)
+{
+    struct Map_info map;
+    static struct line_pnts *Points = NULL;
+    struct line_cats *Cats = NULL;
+    geopoint *top, *gpt, *prev;
+    int np, ltype, eof;
+    struct Cell_head wind;
+    int ndim;
+    const char *mapset;
+
+    np = 0;
+    eof = 0;
+    
+    mapset = G_find_vector2(name, "");
+    if (!mapset) {
+	G_warning(_("Vector map <%s> not found"), name);
+	return NULL;
+    }
+    
+    Vect_set_open_level(1);
+    if (Vect_open_old(&map, name, "") == -1) {
+	G_fatal_error(_("Unable to open vector map <%s>"),
+		      G_fully_qualified_name(name, mapset));
+    }
+    
+    Points = Vect_new_line_struct();
+    Cats = Vect_new_cats_struct();
+    
+    top = gpt = (geopoint *) G_malloc(sizeof(geopoint));
+    G_zero(gpt, sizeof(geopoint));
+    if (!top) {
+	return NULL;
+    }
+
+    G_get_set_window(&wind);
+    Vect_set_constraint_region(&map, wind.north, wind.south, wind.east,
+			       wind.west, PORT_DOUBLE_MAX, -PORT_DOUBLE_MAX);
+
+    /* get ndim */
+    *has_z = 0;
+    ndim = 2;
+    if (Vect_is_3d(&map)) {
+	*has_z = 1;
+	ndim = 3;
+    }
+
+    while (eof == 0) {
+	ltype = Vect_read_next_line(&map, Points, Cats);
+	switch (ltype) {
+	case -1:
+	    {
+		G_warning(_("Unable to read vector map <%s>"),
+			  G_fully_qualified_name(name, mapset));
+		return NULL;
+	    }
+	case -2:		/* EOF */
+	    {
+		eof = 1;
+		continue;
+	    }
+	}
+	if ((ltype & GV_POINTS)) {
+	    np++;
+	    gpt->p3[X] = Points->x[0];
+	    gpt->p3[Y] = Points->y[0];
+
+	    if (ndim > 2) {
+		gpt->dims = 3;
+		gpt->p3[Z] = Points->z[0];
+	    }
+	    else {
+		gpt->dims = 2;
+	    }
+
+	    /* Store category info for thematic display */
+	    if (Cats->n_cats > 0) {
+		gpt->cats = Cats;
+		Cats = Vect_new_cats_struct();
+	    }
+	    else {
+		Vect_reset_cats(Cats);
+	    }
+	    /* initialize style */
+	    gpt->highlighted = 0;
+	    
+	    G_debug(5, "loading vector point %d x=%f y=%f ncats=%d",
+		    np, Points->x[0], Points->y[0], Cats->n_cats);
+
+	    gpt->next = (geopoint *) G_malloc(sizeof(geopoint));	/* G_fatal_error */
+	    G_zero(gpt->next, sizeof(geopoint));
+	    if (!gpt->next) {
+		return NULL;
+	    }
+
+	    prev = gpt;
+	    gpt = gpt->next;
+	}
+
+    }
+    if (np > 0) {
+	prev->next = NULL;
+	G_free(gpt);
+    }
+
+    Vect_close(&map);
+
+    if (!np) {
+	G_warning(_("No points from vector map <%s> fall within current region"),
+		  G_fully_qualified_name(name, mapset));
+	return (NULL);
+    }
+    else {
+	G_message(_("Vector map <%s> loaded (%d points)"),
+		  G_fully_qualified_name(name, mapset), np);
+    }
+
+    *nsites = np;
+
+    return top;
+}
+
+/*!
+  \brief Load styles for geopoints based on thematic mapping
+
+  \param gp pointer to geosite structure
+  \param colors pointer to Colors structure or NULL
+  
+  \return number of points defined by thematic mapping
+  \return -1 on error
+*/
+int Gp_load_sites_thematic(geosite *gp, struct Colors *colors)
+{
+    geopoint *gpt;
+
+    struct Map_info Map;
+    struct field_info *Fi;
+    
+    int nvals, cat, npts, nskipped;
+    int red, blu, grn;
+    const char *str;
+    const char *mapset;
+
+    dbDriver *driver;
+    dbValue value;
+    
+    if(!gp || !gp->tstyle || !gp->filename)
+	return -1;
+
+    mapset = G_find_vector2(gp->filename, "");
+    if (!mapset) {
+	G_fatal_error(_("Vector map <%s> not found"), gp->filename);
+    }
+    
+    Vect_set_open_level(1);
+    if (Vect_open_old(&Map, gp->filename, "") == -1) {
+	G_fatal_error(_("Unable to open vector map <%s>"),
+		      G_fully_qualified_name(gp->filename, mapset));
+    }
+    
+    Fi = Vect_get_field(&Map, gp->tstyle->layer);
+    if (!Fi) {
+	G_warning(_("Database connection not defined for layer %d"),
+		  gp->tstyle->layer);
+    }
+    else {
+	driver = db_start_driver_open_database(Fi->driver, Fi->database);
+	if (!driver)
+	    G_fatal_error(_("Unable to open database <%s> by driver <%s>"),
+			  Fi->database, Fi->driver);
+    }
+    G_message(_("Loading thematic points layer <%s>..."),
+	      G_fully_qualified_name(gp->filename, mapset));
+    npts = nskipped = 0;
+    for(gpt = gp->points; gpt; gpt = gpt->next) {
+	gpt->style = (gvstyle *) G_malloc(sizeof(gvstyle));
+	G_zero(gpt->style, sizeof(gvstyle));
+	
+	/* use default style */
+	gpt->style->color  = gp->style->color;
+	gpt->style->symbol = gp->style->symbol;
+	gpt->style->size   = gp->style->size;
+	gpt->style->width  = gp->style->width;
+	
+	cat = -1;
+	if (gpt->cats)
+	    Vect_cat_get(gpt->cats, gp->tstyle->layer, &cat);
+	if (cat < 0) {
+	    nskipped++;
+	    continue;
+	}
+
+	/* color */
+	if (colors) {
+	    if (!Rast_get_c_color((const CELL *) &cat, &red, &grn, &blu, colors)) {
+		G_warning(_("No color rule defined for category %d"), cat);
+		gpt->style->color = gp->style->color;
+	    }
+	    gpt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
+		((int)((blu) << 16) & BLU_MASK);
+	}
+	if (gp->tstyle->color_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->color_column, &value);
+	    if (nvals < 1)
+		continue;
+	    str = db_get_value_string(&value);
+	    if (!str)
+		continue;
+	    if (G_str_to_color(str, &red, &grn, &blu) != 1) {
+		G_warning(_("Invalid color definition (%s)"),
+			  str);
+		gpt->style->color = gp->style->color;
+	    }
+	    else {
+		gpt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
+		    ((int)((blu) << 16) & BLU_MASK);
+	    }
+	}
+	
+	/* size */
+	if (gp->tstyle->size_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->size_column, &value);
+	    if (nvals < 1)
+		continue;
+	    gpt->style->size = db_get_value_int(&value);
+	}
+
+	/* width */
+	if (gp->tstyle->width_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->width_column, &value);
+	    if (nvals < 1)
+		continue;
+	    gpt->style->width = db_get_value_int(&value);
+	}
+
+	/* symbol/marker */
+	if (gp->tstyle->symbol_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gp->tstyle->symbol_column, &value);
+	    if (nvals < 1)
+		continue;
+	    str = db_get_value_string(&value);
+	    gpt->style->symbol = GP_str_to_marker(str);
+	}
+	
+	npts++;
+    }
+    
+    if (nskipped > 0)
+	G_warning(_("%d points without category. "
+		    "Unable to determine color rules for features without category."),
+		  nskipped);
+    return npts;
+}

Copied: grass/trunk/lib/ogsf/gs2.c (from rev 62429, grass/trunk/lib/ogsf/GS2.c)
===================================================================
--- grass/trunk/lib/ogsf/gs2.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gs2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,3495 @@
+/*!
+   \file GS2.c
+
+   \brief OGSF library - loading and manipulating surfaces (higher level functions)
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   Plans for handling color maps:
+   NOW:
+   if able to load as unsigned char, make lookup table containing palette
+   otherwise, load directly as packed color, set lookup = NULL
+   MAYBE LATER:
+   if able to load as POSITIVE short, make lookup table containing palette
+   - may want to calculate savings first (ie,  numcells > 32768)
+   (not exactly, it's Friday & time to go home - figure it later)
+   otherwise, load directly as packed color, set lookup = NULL
+   MESSY! - need to fix up!
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown USACERL (1993)
+   \author Pierre de Mouveaux <p_de_mouveaux hotmail.com> (updated October 1999)
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <math.h>
+
+#include <grass/config.h>
+
+#if defined(OPENGL_X11) || defined(OPENGL_WINDOWS)
+#include <GL/gl.h>
+#include <GL/glu.h>
+#elif defined(OPENGL_AQUA)
+#include <OpenGL/gl.h>
+#include <OpenGL/glu.h>
+#endif
+
+#include <grass/gis.h>
+#include <grass/raster.h>
+#include <grass/ogsf.h>
+#include <grass/glocale.h>
+
+#include "gsget.h"
+#include "rowcol.h"
+#include "rgbpack.h"
+
+/* Hack to make NVIZ2.2 query functions.("What's Here" and "Look at")
+ * to work.
+ * Uses gs_los_intersect1() instead of gs_los_intersect().
+ * Pierre de Mouveaux - 31 oct. 1999. p_de_mouveaux at hotmail.com.
+ */
+#define NVIZ_HACK 1
+
+int gsd_getViewport(GLint *, GLint *);
+
+/* array of surface ids */
+static int Surf_ID[MAX_SURFS];
+static int Next_surf = 0;
+static int SDref_surf = 0;
+
+/* attributes array */
+static float Default_const[MAX_ATTS];
+static float Default_nulls[MAX_ATTS];
+
+/* largest dimension */
+static float Longdim;
+
+/* N, S, W, E */
+static float Region[4];
+static geoview Gv;
+static geodisplay Gd;
+static struct Cell_head wind;
+static int Buffermode;
+static int Numlights = 0;
+static int Resetlight = 1;
+static int Modelshowing = 0;
+
+void void_func(void)
+{
+    return;
+}
+
+/*!
+   \brief Initialize OGSF library
+
+   Get region settings - wind
+
+   Set Region (NSWE array) and compute scale
+ */
+void GS_libinit(void)
+{
+    static int first = 1;
+
+    G_get_set_window(&wind);
+
+    Region[0] = wind.north;
+    Region[1] = wind.south;
+    Region[2] = wind.west;
+    Region[3] = wind.east;
+
+    /* scale largest dimension to GS_UNIT_SIZE */
+    if ((wind.east - wind.west) > (wind.north - wind.south)) {
+	Longdim = (wind.east - wind.west);
+    }
+    else {
+	Longdim = (wind.north - wind.south);
+    }
+
+    Gv.scale = GS_UNIT_SIZE / Longdim;
+
+    G_debug(1, "GS_libinit(): n=%f s=%f w=%f e=%f scale=%f first=%d",
+	    Region[0], Region[1], Region[2], Region[3], Gv.scale, first);
+    
+    Cxl_func = void_func;
+    Swap_func = void_func;
+
+    
+    if (first) {
+	gs_init();
+    }
+
+    first = 0;
+
+    return;
+}
+
+/*!
+   \brief Get largest dimension
+
+   \param[out] dim dimension
+
+   \return 1
+ */
+int GS_get_longdim(float *dim)
+{
+    *dim = Longdim;
+
+    G_debug(3, "GS_get_longdim(): dim=%g", *dim);
+
+    return (1);
+}
+
+/*!
+   \brief Get 2D region extent
+
+   \param[out] n,s,w,e extent values
+
+   \return 1
+ */
+int GS_get_region(float *n, float *s, float *w, float *e)
+{
+    *n = Region[0];
+    *s = Region[1];
+    *w = Region[2];
+    *e = Region[3];
+
+    return (1);
+}
+
+/*!
+   \brief Set default attributes for map objects
+
+   \param defs attributes array (dim MAX_ATTS)
+   \param null_defs null attributes array (dim MAX_ATTS)
+ */
+void GS_set_att_defaults(float *defs, float *null_defs)
+{
+    int i;
+
+    G_debug(3, "GS_set_att_defaults");
+
+    for (i = 0; i < MAX_ATTS; i++) {
+	Default_const[i] = defs[i];
+	Default_nulls[i] = null_defs[i];
+    }
+
+    return;
+}
+
+/*!
+   Check if surface exists
+
+   \param id surface id
+
+   \return 0 not found
+   \return 1 found
+ */
+int GS_surf_exists(int id)
+{
+    int i, found = 0;
+
+    G_debug(3, "GS_surf_exists(): id=%d", id);
+
+
+    if (NULL == gs_get_surf(id)) {
+	return (0);
+    }
+
+    for (i = 0; i < Next_surf && !found; i++) {
+	if (Surf_ID[i] == id) {
+	    found = 1;
+	}
+    }
+
+    return (found);
+}
+
+/*!
+   \brief Add new surface
+
+   Note that origin has 1/2 cell added to represent center of cells
+   because library assumes that east - west = (cols - 1) * ew_res,
+   since left and right columns are on the edges.
+
+   \return surface id
+   \return -1 on error (MAX_SURFS exceded)
+ */
+int GS_new_surface(void)
+{
+    geosurf *ns;
+
+    G_debug(3, "GS_new_surface():");
+
+    if (Next_surf < MAX_SURFS) {
+	ns = gs_get_new_surface();
+	gs_init_surf(ns, wind.west + wind.ew_res / 2.,
+		     wind.south + wind.ns_res / 2., wind.rows, wind.cols,
+		     wind.ew_res, wind.ns_res);
+	gs_set_defaults(ns, Default_const, Default_nulls);
+
+	/* make default shine current */
+	gs_set_att_src(ns, ATT_SHINE, CONST_ATT);
+
+	Surf_ID[Next_surf] = ns->gsurf_id;
+	++Next_surf;
+
+	G_debug(3, "    id=%d", ns->gsurf_id);
+
+	return (ns->gsurf_id);
+    }
+
+
+
+    return (-1);
+}
+void GS_set_light_reset(int i)
+{
+    Resetlight = i;
+    if (i)
+	Numlights = 0;
+}
+int GS_get_light_reset(void)
+{
+    return Resetlight;
+}
+/*!
+   \brief Add new model light
+
+   \return light model id
+   \return -1 on error (MAX_LIGHTS exceded)
+ */
+int GS_new_light(void)
+{
+    int i;
+
+    if (GS_get_light_reset()) {
+
+	GS_set_light_reset(0);
+
+	for (i = 0; i < MAX_LIGHTS; i++) {
+	    Gv.lights[i].position[X] = Gv.lights[i].position[Y] = 0.0;
+	    Gv.lights[i].position[Z] = 1.0;
+	    Gv.lights[i].position[W] = 0.0;	/* infinite */
+	    Gv.lights[i].color[0] = Gv.lights[i].color[1] =
+		Gv.lights[i].color[2] = 1.0;
+	    Gv.lights[i].ambient[0] = Gv.lights[i].ambient[1] =
+		Gv.lights[i].ambient[2] = 0.2;
+	    Gv.lights[i].shine = 32.0;
+	}
+
+	gsd_init_lightmodel();
+    }
+
+    if (Numlights < MAX_LIGHTS) {
+	gsd_deflight(Numlights + 1, &(Gv.lights[Numlights]));
+	gsd_switchlight(Numlights + 1, 1);
+
+	return ++Numlights;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Set light position
+
+   \bug I think lights array doesnt match sgi_light array
+
+   \param num light id (starts with 1)
+   \param xpos,ypos,zpos coordinates (model)
+   \param local local coordinate (for viewport)
+ */
+void GS_setlight_position(int num, float xpos, float ypos, float zpos,
+			  int local)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    Gv.lights[num].position[X] = xpos;
+	    Gv.lights[num].position[Y] = ypos;
+	    Gv.lights[num].position[Z] = zpos;
+	    Gv.lights[num].position[W] = (float)local;
+
+	    gsd_deflight(num + 1, &(Gv.lights[num]));
+	}
+    }
+
+    return;
+}
+
+
+/*!
+   \brief Get light position
+
+   \param num light id (starts at 1)
+   \param[out] xpos,ypos,zpos coordinates
+   \param[out] local ?
+ */
+void GS_getlight_position(int num, float *xpos, float *ypos, float *zpos,
+			  int *local)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    *xpos = Gv.lights[num].position[X];
+	    *ypos = Gv.lights[num].position[Y];
+	    *zpos = Gv.lights[num].position[Z];
+	    *local = (int)Gv.lights[num].position[W];
+
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Set light color
+
+   \param num light id (starts at 1)
+   \param red,green,blue color values (from 0.0 to 1.0)
+ */
+void GS_setlight_color(int num, float red, float green, float blue)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    Gv.lights[num].color[0] = red;
+	    Gv.lights[num].color[1] = green;
+	    Gv.lights[num].color[2] = blue;
+
+	    gsd_deflight(num + 1, &(Gv.lights[num]));
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Get light color
+
+   \param num light id (starts at 1)
+   \param[out] red,green,blue color values
+ */
+void GS_getlight_color(int num, float *red, float *green, float *blue)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    *red = Gv.lights[num].color[0];
+	    *green = Gv.lights[num].color[1];
+	    *blue = Gv.lights[num].color[2];
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Set light ambient
+
+   Red, green, blue from 0.0 to 1.0
+
+   \param num light id (starts at 1)
+   \param red,green,blue color values
+ */
+void GS_setlight_ambient(int num, float red, float green, float blue)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    Gv.lights[num].ambient[0] = red;
+	    Gv.lights[num].ambient[1] = green;
+	    Gv.lights[num].ambient[2] = blue;
+
+	    gsd_deflight(num + 1, &(Gv.lights[num]));
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Get light ambient
+
+   \param num light id (starts at 1)
+   \param[out] red,green,blue color values
+ */
+void GS_getlight_ambient(int num, float *red, float *green, float *blue)
+{
+    if (num) {
+	num -= 1;
+	if (num < Numlights) {
+	    *red = Gv.lights[num].ambient[0];
+	    *green = Gv.lights[num].ambient[1];
+	    *blue = Gv.lights[num].ambient[2];
+	}
+    }
+
+    return;
+}
+
+
+/*!
+   \brief Switch off all lights
+ */
+void GS_lights_off(void)
+{
+    int i;
+
+    for (i = 0; i < Numlights; i++) {
+	gsd_switchlight(i + 1, 0);
+    }
+
+    return;
+}
+
+/*!
+   \brief Switch on all lights
+ */
+void GS_lights_on(void)
+{
+    int i;
+
+    for (i = 0; i < Numlights; i++) {
+	gsd_switchlight(i + 1, 1);
+    }
+
+    return;
+}
+
+/*!
+   \brief Switch on/off light
+
+   \param num light id (starts at 1)
+   \param on non-zero for 'on' otherwise 'off'
+ */
+void GS_switchlight(int num, int on)
+{
+    if (num) {
+	num -= 1;
+
+	if (num < Numlights) {
+	    gsd_switchlight(num + 1, on);
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Check if transparency is set
+
+   \return 0 transparency not set
+   \return 1 transparency is set
+ */
+int GS_transp_is_set(void)
+{
+    return (gs_att_is_set(NULL, ATT_TRANSP) || (FC_GREY == gsd_getfc()));
+}
+
+/*!
+   \brief Retrieves coordinates for lighting model position, at center of view
+
+   \param pos[out] coordinates
+ */
+void GS_get_modelposition1(float pos[])
+{
+    /* TODO: Still needs work to handle other cases */
+    /* this is a quick hack to get lighting adjustments debugged */
+    /*
+       GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], center);
+       GS_v3mult(center, 1000);
+       GS_v3add(center, Gv.from_to[FROM]);
+     */
+
+    gs_get_datacenter(pos);
+    gs_get_data_avg_zmax(&(pos[Z]));
+
+    G_debug(1, "GS_get_modelposition1(): model position: %f %f %f",
+	    pos[X], pos[Y], pos[Z]);
+
+    return;
+}
+
+/*!
+   \brief Retrieves coordinates for lighting model position, at center of view
+
+   Position at nearclip * 2: tried nearclip + siz, but since need to
+   know position to calculate size, have two dependent variables
+   (nearclip * 2) from eye.
+
+   \param siz[out] size
+   \param pos[out] coordinates (X, Y, Z)
+ */
+void GS_get_modelposition(float *siz, float *pos)
+{
+    float dist, near_h, dir[3];
+
+    dist = 2. * Gd.nearclip;
+
+    near_h = 2.0 * tan(4.0 * atan(1.) * Gv.fov / 3600.) * dist;
+    *siz = near_h / 8.0;
+
+    /* prevent clipping - would only happen if fov > ~127 degrees, at
+       fov = 2.0 * atan(2.0) */
+
+    if (*siz > Gd.nearclip) {
+	*siz = Gd.nearclip;
+    }
+
+    GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], dir);
+
+    pos[X] = Gv.from_to[FROM][X] + dir[X] * dist;
+    pos[Y] = Gv.from_to[FROM][Y] + dir[Y] * dist;
+    pos[Z] = Gv.from_to[FROM][Z] + dir[Z] * dist;
+
+    return;
+}
+
+
+/*!
+   \brief Set decoration, north arrow ??
+
+   \todo scale used to calculate len of arrow still needs work
+   needs go function that returns center / eye distance
+   gsd_get_los function is not working correctly ??
+
+   \param pt point value in true world coordinates (?)
+   \param id surface id
+   \param[out] pos2 output coordinates
+ */
+void GS_set_Narrow(int *pt, int id, float *pos2)
+{
+    geosurf *gs;
+    float x, y, z;
+    GLdouble modelMatrix[16], projMatrix[16];
+    GLint viewport[4];
+
+    if (GS_get_selected_point_on_surface(pt[X], pt[Y], &id, &x, &y, &z)) {
+	gs = gs_get_surf(id);
+	if (gs) {
+	    z = gs->zmax;
+	    pos2[X] = (float)x - gs->ox + gs->x_trans;
+	    pos2[Y] = (float)y - gs->oy + gs->y_trans;
+	    pos2[Z] = (float)z + gs->z_trans;
+
+	    return;
+	}
+    }
+    else {
+	gs = gs_get_surf(id);
+
+	/* Need to get model matrix, etc 
+	 * to run gluUnProject
+	 */
+	gsd_pushmatrix();
+	gsd_do_scale(1);
+	glGetDoublev(GL_MODELVIEW_MATRIX, modelMatrix);
+	glGetDoublev(GL_PROJECTION_MATRIX, projMatrix);
+	glGetIntegerv(GL_VIEWPORT, viewport);
+
+	if (gs) {
+	    GLdouble out_near[3], out_far[3];
+	    GLdouble factor;
+	    GLdouble out[3];
+
+	    z = (float)gs->zmax + gs->z_trans;
+
+	    gluUnProject((GLdouble) pt[X], (GLdouble) pt[Y], (GLdouble) 0.,
+			 modelMatrix, projMatrix, viewport,
+			 &out_near[X], &out_near[Y], &out_near[Z]);
+	    gluUnProject((GLdouble) pt[X], (GLdouble) pt[Y], (GLdouble) 1.,
+			 modelMatrix, projMatrix, viewport,
+			 &out_far[X], &out_far[Y], &out_far[Z]);
+
+	    glPopMatrix();
+
+	    factor = (out_near[Z] - z) / (out_near[Z] - out_far[Z]);
+
+	    out[X] = out_near[X] - ((out_near[X] - out_far[X]) * factor);
+	    out[Y] = out_near[Y] - ((out_near[Y] - out_far[Y]) * factor);
+	    out[Z] = z;
+
+	    pos2[X] = (float)out[X];
+	    pos2[Y] = (float)out[Y];
+	    pos2[Z] = (float)out[Z];
+
+	    return;
+
+	}
+    }
+    return;
+}
+
+/*!
+   \brief Draw place marker
+
+   Used to display query point for raster queries.
+
+   \param id surface id
+   \param pt point, X, Y value in true world coordinates
+ */
+void GS_draw_X(int id, float *pt)
+{
+    geosurf *gs;
+    Point3 pos;
+    float siz;
+    gvstyle style;
+
+    if ((gs = gs_get_surf(id))) {
+	GS_get_longdim(&siz);
+	style.size = siz / 200.;
+	pos[X] = pt[X] - gs->ox;
+	pos[Y] = pt[Y] - gs->oy;
+	_viewcell_tri_interp(gs, pos);
+
+	gsd_pushmatrix();
+
+	gsd_do_scale(1);
+	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
+	gsd_linewidth(1);
+
+	if (CONST_ATT == gs_get_att_src(gs, ATT_TOPO)) {
+	    pos[Z] = gs->att[ATT_TOPO].constant;
+	    gs = NULL;		/* tells gpd_obj to use given Z val */
+	}
+	style.color = Gd.bgcol;
+	style.symbol = ST_GYRO;
+	gpd_obj(gs, &style, pos);
+	gsd_flush();
+
+	gsd_popmatrix();
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw line on surface
+
+   \param id surface id
+   \param x1,y1,x2,y2 line nodes
+ */
+void GS_draw_line_onsurf(int id, float x1, float y1, float x2, float y2)
+{
+    float p1[2], p2[2];
+    geosurf *gs;
+
+    if ((gs = gs_get_surf(id))) {
+	p1[X] = x1 - gs->ox;
+	p1[Y] = y1 - gs->oy;
+	p2[X] = x2 - gs->ox;
+	p2[Y] = y2 - gs->oy;
+
+	gsd_pushmatrix();
+
+	gsd_do_scale(1);
+	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
+	gsd_linewidth(1);
+
+	gsd_color_func(GS_default_draw_color());
+	gsd_line_onsurf(gs, p1, p2);
+
+	gsd_popmatrix();
+	gsd_flush();
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw multiline on surface
+
+   Like above but limits points in line to n or points found in segment,
+   whichever is smaller.
+
+   \param id surface id
+   \param x1,y1,x2,y2 line nodes
+
+   \return number of points used
+ */
+int GS_draw_nline_onsurf(int id, float x1, float y1, float x2, float y2,
+			 float *lasp, int n)
+{
+    float p1[2], p2[2];
+    geosurf *gs;
+    int ret = 0;
+
+    if ((gs = gs_get_surf(id))) {
+	p1[X] = x1 - gs->ox;
+	p1[Y] = y1 - gs->oy;
+	p2[X] = x2 - gs->ox;
+	p2[Y] = y2 - gs->oy;
+
+	gsd_pushmatrix();
+
+	gsd_do_scale(1);
+	gsd_translate(gs->x_trans, gs->y_trans, gs->z_trans);
+	gsd_linewidth(1);
+	gsd_color_func(GS_default_draw_color());
+	ret = gsd_nline_onsurf(gs, p1, p2, lasp, n);
+	gsd_surf2real(gs, lasp);
+
+	gsd_popmatrix();
+	gsd_flush();
+    }
+
+    return (ret);
+}
+
+/*!
+   \brief Draw flow-line on surace
+
+   This is slow - should be moved to gs_ but GS_ good for testing
+   and useful for app programmer
+
+   \param id surface id
+   \param x,y coordinates of flow-line
+ */
+void GS_draw_flowline_at_xy(int id, float x, float y)
+{
+    geosurf *gs;
+    float nv[3], pdir[2], mult;
+    float p1[2], p2[2], next[2];
+    int i = 0;
+
+    if ((gs = gs_get_surf(id))) {
+	p1[X] = x;
+	p1[Y] = y;
+	/* multiply by 1.5 resolutions to ensure a crossing ? */
+	mult = .1 * (VXRES(gs) > VYRES(gs) ? VXRES(gs) : VYRES(gs));
+
+	GS_coordpair_repeats(p1, p1, 50);
+
+	while (1 == GS_get_norm_at_xy(id, p1[X], p1[Y], nv)) {
+	    if (nv[Z] == 1.0) {
+		if (pdir[X] == 0.0 && pdir[Y] == 0.0) {
+		    break;
+		}
+
+		p2[X] = p1[X] + (pdir[X] * mult);
+		p2[Y] = p1[Y] + (pdir[Y] * mult);
+	    }
+	    else {
+		/* use previous direction */
+		GS_v2norm(nv);
+		p2[X] = p1[X] + (nv[X] * mult);
+		p2[Y] = p1[Y] + (nv[Y] * mult);
+		pdir[X] = nv[X];
+		pdir[Y] = nv[Y];
+	    }
+
+	    if (i > 2000) {
+		break;
+	    }
+
+	    if (GS_coordpair_repeats(p1, p2, 0)) {
+		break;
+	    }
+
+	    /* Think about this: */
+	    /* degenerate line means edge or level edge ? */
+	    /* next is filled with last point drawn */
+	    if (2 > GS_draw_nline_onsurf(id, p1[X], p1[Y],
+					 p2[X], p2[Y], next, 3)) {
+		break;
+	    }
+
+	    p1[X] = next[X];
+	    p1[Y] = next[Y];
+	}
+
+	G_debug(3, "GS_draw_flowline_at_xy(): dir: %f %f", nv[X], nv[Y]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw fringe around data (surface) at selected corners
+
+   \param id surface id
+   \param clr color
+   \param elev elevation value
+   \param where nw/ne/sw/se edges - 0 (turn off) 1 (turn on)
+ */
+void GS_draw_fringe(int id, unsigned long clr, float elev, int *where)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_draw_fringe(): id: %d clr: %ld elev %f edges: %d %d %d %d",
+	    id, clr, elev, where[0], where[1], where[2], where[3]);
+    if ((gs = gs_get_surf(id)))
+	gsd_display_fringe(gs, clr, elev, where);
+
+}
+
+
+/*!
+   \brief Draw legend
+
+   \todo add legend from list option
+   make font loading more flexible
+
+   \param name legend name
+   \param fontbase font-base
+   \param size ? 
+   \param flags legend flags
+   \param range values range
+   \param pt ?
+ */
+int GS_draw_legend(const char *name, GLuint fontbase, int size, int *flags,
+		   float *range, int *pt)
+{
+    int list_no;
+
+    list_no = gsd_put_legend(name, fontbase, size, flags, range, pt);
+
+    return (list_no);
+}
+
+/*!
+   \brief Draw pre-defined list
+
+   Uses glFlush() to ensure all drawing is complete
+   before returning
+
+   \param list_id list id
+ */
+void GS_draw_list(GLuint list_id)
+{
+    gsd_calllist(list_id);
+    glFlush();
+    return;
+}
+
+/*!
+   \brief Draw all glLists
+
+   Uses glFlush() to ensure all drawing is complete
+   before returning
+ */
+void GS_draw_all_list(void)
+{
+    gsd_calllists(0);		/* not sure if 0 is right - MN */
+    glFlush();
+    return;
+}
+
+/*!
+   \brief Delete pre-defined list
+
+   \param list_id list id
+ */
+void GS_delete_list(GLuint list_id)
+{
+    gsd_deletelist(list_id, 1);
+
+    return;
+}
+
+/*!
+   \brief Draw lighting model
+ */
+void GS_draw_lighting_model1(void)
+{
+    static float center[3];
+    float tcenter[3];
+
+    if (!Modelshowing) {
+	GS_get_modelposition1(center);
+    }
+
+    GS_v3eq(tcenter, center);
+
+    gsd_zwritemask(0x0);
+    gsd_backface(1);
+
+    gsd_colormode(CM_AD);
+    gsd_shademodel(DM_GOURAUD);
+    gsd_pushmatrix();
+    gsd_do_scale(1);
+
+    if (Gv.vert_exag) {
+	tcenter[Z] *= Gv.vert_exag;
+	gsd_scale(1.0, 1.0, 1. / Gv.vert_exag);
+    }
+
+    gsd_drawsphere(tcenter, 0xDDDDDD, (float)(Longdim / 10.));
+    gsd_popmatrix();
+    Modelshowing = 1;
+
+    gsd_backface(0);
+    gsd_zwritemask(0xffffffff);
+
+    return;
+}
+
+/*!
+   \brief Draw lighting model
+
+   Just turn off any cutting planes and draw it just outside near
+   clipping plane, since lighting is infinite now
+ */
+void GS_draw_lighting_model(void)
+{
+    static float center[3], size;
+    float tcenter[3], tsize;
+    int i, wason[MAX_CPLANES];
+
+    gsd_get_cplanes_state(wason);
+
+    for (i = 0; i < MAX_CPLANES; i++) {
+	if (wason[i]) {
+	    gsd_cplane_off(i);
+	}
+    }
+
+
+    if (!Modelshowing) {
+	GS_get_modelposition(&size, center);
+    }
+
+    GS_v3eq(tcenter, center);
+    tsize = size;
+
+    gsd_zwritemask(0x0);
+    gsd_backface(1);
+
+    gsd_colormode(CM_DIFFUSE);
+    gsd_shademodel(DM_GOURAUD);
+    gsd_pushmatrix();
+    gsd_drawsphere(tcenter, 0xDDDDDD, tsize);
+    gsd_popmatrix();
+    Modelshowing = 1;
+
+    gsd_backface(0);
+    gsd_zwritemask(0xffffffff);
+
+    for (i = 0; i < MAX_CPLANES; i++) {
+	if (wason[i]) {
+	    gsd_cplane_on(i);
+	}
+    }
+
+    gsd_flush();
+
+    return;
+}
+
+/*!
+   \brief Update current mask
+
+   May be called to update total mask for a surface at convenient times
+   instead of waiting until ready to redraw surface
+
+   \param id surface id
+
+   \return ?
+ */
+int GS_update_curmask(int id)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+    return (gs_update_curmask(gs));
+}
+
+/*!
+   \brief Check if point is masked ?
+
+   \param id surface id
+   \param pt point
+
+   \return 1 masked
+   \return 0 not masked
+   \return -1 on error, invalid surface id
+ */
+int GS_is_masked(int id, float *pt)
+{
+    geosurf *gs;
+    Point3 tmp;
+
+    if ((gs = gs_get_surf(id))) {
+	tmp[X] = pt[X] - gs->ox;
+	tmp[Y] = pt[Y] - gs->oy;
+
+	return (gs_point_is_masked(gs, tmp));
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Unset Scaled Difference surface
+ */
+void GS_unset_SDsurf(void)
+{
+    gsdiff_set_SDref(NULL);
+    SDref_surf = 0;
+
+    return;
+}
+
+/*!
+   \brief Set surface as Scaled Difference surface
+
+   \param id surface id
+
+   \return 1 on success
+   \return 0 on error, invalid surface id
+ */
+int GS_set_SDsurf(int id)
+{
+    geosurf *gs;
+
+    if ((gs = gs_get_surf(id))) {
+	gsdiff_set_SDref(gs);
+	SDref_surf = id;
+
+	return (1);
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Set ?
+
+   \param scale scale value
+
+   \return 1
+ */
+int GS_set_SDscale(float scale)
+{
+    gsdiff_set_SDscale(scale);
+
+    return (1);
+}
+
+/*!
+   \brief Get ?
+
+   \param[out] id ?
+
+   \return 1 on success
+   \return 0 on error
+ */
+int GS_get_SDsurf(int *id)
+{
+    geosurf *gs;
+
+    if ((gs = gsdiff_get_SDref())) {
+	*id = SDref_surf;
+
+	return (1);
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Get ?
+
+   \param[out] scale value
+
+   \return 1
+ */
+int GS_get_SDscale(float *scale)
+{
+    *scale = gsdiff_get_SDscale();
+
+    return (1);
+}
+
+/*!
+   \brief Update normals
+
+   \param id surface id
+
+   \return ?
+ */
+int GS_update_normals(int id)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    return (gs_calc_normals(gs));
+}
+
+/*!
+   \brief Get attributes
+
+   \param id surface id
+   \param att
+   \param[out] set
+   \param[out] constant
+   \param[out] mapname
+
+   \return 1 on success
+   \return -1 on error (invalid surface id)
+ */
+int GS_get_att(int id, int att, int *set, float *constant, char *mapname)
+{
+    int src;
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+    if (gs) {
+	if (-1 != (src = gs_get_att_src(gs, att))) {
+	    *set = src;
+
+	    if (src == CONST_ATT) {
+		*constant = gs->att[att].constant;
+	    }
+	    else if (src == MAP_ATT) {
+		strcpy(mapname, gsds_get_name(gs->att[att].hdata));
+	    }
+
+	    return (1);
+	}
+
+	return (-1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get surface category on given position
+
+   Prints "no data" or a description (i.e., "coniferous forest") to
+   <i>catstr</i>. Usually call after GS_get_selected_point_on_surface().
+   Define <i>att</i> as MAP_ATT
+
+   \todo Allocate catstr using G_store()
+   
+   \param id surface id
+   \param att attribute id (MAP_ATT)
+   \param catstr cat string (must be allocated, dim?)
+   \param x,y real coordinates
+
+   \return -1 if no category info or point outside of window
+   \return 1 on success
+*/
+int GS_get_cat_at_xy(int id, int att, char *catstr, float x, float y)
+{
+    int offset, drow, dcol, vrow, vcol;
+    float ftmp, pt[3];
+    typbuff *buff;
+    geosurf *gs;
+
+    *catstr = '\0';
+    gs = gs_get_surf(id);
+
+    if (NULL == gs) {
+	return -1;
+    }
+
+    pt[X] = x;
+    pt[Y] = y;
+
+    gsd_real2surf(gs, pt);
+    if (gs_point_is_masked(gs, pt)) {
+	return -1;
+    }
+
+    if (!in_vregion(gs, pt)) {
+	return -1;
+    }
+
+    if (MAP_ATT != gs_get_att_src(gs, att)) {
+	sprintf(catstr, _("no category info"));
+	return -1;
+    }
+
+    buff = gs_get_att_typbuff(gs, att, 0);
+
+    vrow = Y2VROW(gs, pt[Y]);
+    vcol = X2VCOL(gs, pt[X]);
+    drow = VROW2DROW(gs, vrow);
+    dcol = VCOL2DCOL(gs, vcol);
+
+    offset = DRC2OFF(gs, drow, dcol);
+    
+    if (GET_MAPATT(buff, offset, ftmp)) {
+	return
+	    (Gs_get_cat_label(gsds_get_name(gs->att[att].hdata),
+			      drow, dcol, catstr));
+    }
+
+    sprintf(catstr, _("no data"));
+
+    return 1;
+}
+
+/*!
+   \brief Get surface normal at x,y (real coordinates)
+
+   Usually call after GS_get_selected_point_on_surface()
+
+   \param id surface id
+   \param x,y real coordinates
+   \param[out] nv surface normal
+
+   \return -1 if point outside of window or masked
+   \return 1 on success
+ */
+int GS_get_norm_at_xy(int id, float x, float y, float *nv)
+{
+    int offset, drow, dcol, vrow, vcol;
+    float pt[3];
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (NULL == gs) {
+	return (-1);
+    }
+
+    if (gs->norm_needupdate) {
+	gs_calc_normals(gs);
+    }
+
+    pt[X] = x;
+    pt[Y] = y;
+
+    gsd_real2surf(gs, pt);
+    if (gs_point_is_masked(gs, pt)) {
+	return (-1);
+    }
+
+    if (!in_vregion(gs, pt)) {
+	return (-1);
+    }
+
+    vrow = Y2VROW(gs, pt[Y]);
+    vcol = X2VCOL(gs, pt[X]);
+    drow = VROW2DROW(gs, vrow);
+    dcol = VCOL2DCOL(gs, vcol);
+
+    offset = DRC2OFF(gs, drow, dcol);
+
+    if (gs->norms) {
+	FNORM(gs->norms[offset], nv);
+    }
+    else {
+	/* otherwise must be a constant */
+	nv[0] = 0.0;
+	nv[1] = 0.0;
+	nv[2] = 1.0;
+    }
+
+    return (1);
+}
+
+/*!
+   \brief Get RGB color at given point
+
+   Colors are translated to rgb and returned as Rxxx Gxxx Bxxx Usually
+   call after GS_get_selected_point_on_surface().
+
+   Prints NULL or the value (i.e., "921.5") to valstr
+
+   \param id surface id
+   \param att attribute id
+   \param[out] valstr value string (allocated, dim?)
+   \param x,y real coordinates
+   
+   \return -1 if point outside of window or masked
+   \return 1 on success
+ */
+int GS_get_val_at_xy(int id, int att, char *valstr, float x, float y)
+{
+    int offset, drow, dcol, vrow, vcol;
+    float ftmp, pt[3];
+    typbuff *buff;
+    geosurf *gs;
+
+    *valstr = '\0';
+    gs = gs_get_surf(id);
+    
+    if (NULL == gs) {
+	return -1;
+    }
+
+    pt[X] = x;
+    pt[Y] = y;
+
+    gsd_real2surf(gs, pt);
+
+    if (gs_point_is_masked(gs, pt)) {
+	return -1;
+    }
+
+    if (!in_vregion(gs, pt)) {
+	return (-1);
+    }
+
+    if (CONST_ATT == gs_get_att_src(gs, att)) {
+	if (att == ATT_COLOR) {
+	    int r, g, b, i;
+
+	    i = gs->att[att].constant;
+	    sprintf(valstr, "R%d G%d B%d",
+		    INT_TO_RED(i, r), INT_TO_GRN(i, g), INT_TO_BLU(i, b));
+	}
+	else {
+	    sprintf(valstr, "%f", gs->att[att].constant);
+	}
+
+	return 1;
+    }
+    else if (MAP_ATT != gs_get_att_src(gs, att)) {
+	return -1;
+    }
+
+    buff = gs_get_att_typbuff(gs, att, 0);
+
+    vrow = Y2VROW(gs, pt[Y]);
+    vcol = X2VCOL(gs, pt[X]);
+    drow = VROW2DROW(gs, vrow);
+    dcol = VCOL2DCOL(gs, vcol);
+
+    offset = DRC2OFF(gs, drow, dcol);
+
+    if (GET_MAPATT(buff, offset, ftmp)) {
+	if (att == ATT_COLOR) {
+	    int r, g, b, i;
+
+	    i = gs_mapcolor(gs_get_att_typbuff(gs, ATT_COLOR, 0),
+			    &(gs->att[ATT_COLOR]), offset);
+	    sprintf(valstr, "R%d G%d B%d",
+		    INT_TO_RED(i, r), INT_TO_GRN(i, g), INT_TO_BLU(i, b));
+	}
+	else {
+	    sprintf(valstr, "%f", ftmp);
+	}
+
+	return (1);
+    }
+
+    sprintf(valstr, "NULL");
+
+    return (1);
+}
+
+/*!
+   \brief Unset attribute
+
+   \param id surface id
+   \param att attribute id
+
+   \return ?
+ */
+int GS_unset_att(int id, int att)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+    gs->mask_needupdate = 1;
+
+    return (gs_set_att_src(gs, att, NOTSET_ATT));
+}
+
+/*!
+   \brief Set attribute constant
+
+   \param id surface id
+   \param att attribute id
+   \param constant value
+
+   \return ?
+ */
+int GS_set_att_const(int id, int att, float constant)
+{
+    geosurf *gs;
+    int ret;
+
+    gs = gs_get_surf(id);
+    ret = (gs_set_att_const(gs, att, constant));
+
+    Gs_update_attrange(gs, att);
+
+    return (ret);
+}
+
+/*!
+   \brief Set mask mode
+
+   Mask attribute special: constant is set to indicate invert or no
+
+   \param id surface id
+   \param mode id
+
+   \return mode id
+   \return -1 on error (invalid surface id)
+ */
+int GS_set_maskmode(int id, int mode)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	gs->att[ATT_MASK].constant = mode;
+	gs->mask_needupdate = 1;
+
+	return (mode);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get mask mode
+
+   \param id surface id
+   \param[out] mode id
+
+   \return 1 on success
+   \return -1 on error (invalid surface id)
+ */
+int GS_get_maskmode(int id, int *mode)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*mode = gs->att[ATT_MASK].constant;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set client data
+
+   \param id surface id
+   \param clientd pointer to client data struct
+
+   \return 1 on success
+   \return -1 on error (invalid surface id)
+ */
+int GS_Set_ClientData(int id, void *clientd)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+    if (gs) {
+	gs->clientdata = clientd;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get client data
+
+   \param id surface id
+
+   \return pointer to client data
+   \return NULL on error
+ */
+void *GS_Get_ClientData(int id)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+    if (gs) {
+	return (gs->clientdata);
+    }
+
+    return (NULL);
+}
+
+/*!
+   \brief Get number of surfaces
+
+   \return number of surfaces
+ */
+int GS_num_surfs(void)
+{
+    return (gs_num_surfaces());
+}
+
+/*!
+   \brief Get surface list
+
+   Must be freed when not neeed!
+
+   \param[out] numsurf number of available surfaces
+
+   \return pointer to surface array
+   \return NULL on error
+ */
+int *GS_get_surf_list(int *numsurfs)
+{
+    int i, *ret;
+
+    *numsurfs = Next_surf;
+
+    if (Next_surf) {
+	ret = (int *)G_malloc(Next_surf * sizeof(int));
+
+	for (i = 0; i < Next_surf; i++) {
+	    ret[i] = Surf_ID[i];
+	}
+
+	return (ret);
+    }
+
+    return (NULL);
+}
+
+/*!
+   \brief Delete surface
+
+   \param id surface id
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GS_delete_surface(int id)
+{
+    int i, j, found;
+    
+    found = FALSE;
+    
+    G_debug(1, "GS_delete_surface(): id=%d", id);
+    
+    if (GS_surf_exists(id)) {
+	gs_delete_surf(id);
+	for (i = 0; i < Next_surf && !found; i++) {
+	    if (Surf_ID[i] == id) {
+		found = TRUE;
+
+		for (j = i; j < Next_surf; j++) {
+		    Surf_ID[j] = Surf_ID[j + 1];
+		}
+	    }
+	}
+	
+	gv_update_drapesurfs();
+
+	if (found) {
+	    --Next_surf;
+	    return 1;
+	}
+    }
+
+    return -1;
+}
+
+
+/*!
+   \brief Load raster map as attribute
+
+   \param id surface id
+   \param filename filename
+   \param att attribute descriptor
+
+   \return -1 on error (invalid surface id)
+   \return ?
+ */
+int GS_load_att_map(int id, const char *filename, int att)
+{
+    geosurf *gs;
+    unsigned int changed;
+    unsigned int atty;
+    const char *mapset;
+    struct Cell_head rast_head;
+    int reuse, begin, hdata, ret, neg, has_null;
+    typbuff *tbuff;
+
+    G_debug(3, "GS_load_att_map(): map=%s", filename);
+
+    reuse = ret = neg = has_null = 0;
+    gs = gs_get_surf(id);
+
+    if (NULL == gs) {
+	return -1;
+    }
+
+    gs->mask_needupdate = (ATT_MASK == att || ATT_TOPO == att ||
+			   (gs->nz_topo && ATT_TOPO == att) ||
+			   (gs->nz_color && ATT_COLOR == att));
+
+    gs_set_att_src(gs, att, MAP_ATT);
+
+    /* Check against maps already loaded in memory   */
+    /* if to be color attribute:
+       - if packed color for another surface, OK to reuse
+       - if unchanged, ok to reuse IF it's of type char (will have lookup)
+     */
+    begin = hdata = 1;
+
+    /* Get MAPSET to ensure names are fully qualified */
+    mapset = G_find_raster2(filename, "");
+    if (mapset == NULL) {
+	/* Check for valid filename */
+	G_warning("Raster map <%s> not found", filename);
+	return -1;
+    }
+    
+    /* Check to see if map is in Region */
+    Rast_get_cellhd(filename, mapset, &rast_head);
+    if (rast_head.north <= wind.south ||
+	rast_head.south >= wind.north ||
+	rast_head.east <= wind.west || rast_head.west >= wind.east) {
+
+	G_warning(_("Raster map <%s> is outside of current region. Load failed."),
+		  G_fully_qualified_name(filename, mapset));
+    }
+
+    while (!reuse && (0 < hdata)) {
+	changed = CF_COLOR_PACKED;
+	atty = ATTY_FLOAT | ATTY_CHAR | ATTY_INT | ATTY_SHORT | ATTY_MASK;
+
+	if (0 < (hdata = gsds_findh(filename, &changed, &atty, begin))) {
+
+	    G_debug(3, "GS_load_att_map(): %s already has data handle %d.CF=%x",
+		    filename, hdata, changed);
+
+	    /* handle found */
+	    if (ATT_COLOR == att) {
+		if ((changed == CF_COLOR_PACKED) ||
+		    (!changed && atty == ATTY_CHAR)) {
+		    reuse = 1;
+		}
+	    }
+	    else if (atty == ATTY_MASK && att != ATT_MASK) {
+		reuse = 0;
+		/* should also free mask data & share new - but need backward
+		   reference? */
+	    }
+	    else if (!changed) {
+		reuse = 1;
+	    }
+	}
+
+	begin = 0;
+    }
+
+    if (reuse) {
+	gs->att[att].hdata = hdata;
+	gs_set_att_type(gs, att, atty);	/* ?? */
+
+	/* free lookup  & set to NULL! */
+	if (atty == ATTY_INT) {
+	    if (gs->att[att].lookup) {
+		free(gs->att[att].lookup);
+		gs->att[att].lookup = NULL;
+	    }
+	}
+	/* TODO: FIX THIS stuff with lookup sharing! */
+
+	G_debug(3, "GS_load_att_map(): %s is being reused. hdata=%d",
+		filename, hdata);
+    }
+    else {
+	G_debug(3, "GS_load_att_map(): %s not loaded in correct form - loading now",
+		filename);
+
+	/* not loaded - need to get new dataset handle */
+	gs->att[att].hdata = gsds_newh(filename);
+
+	tbuff = gs_get_att_typbuff(gs, att, 1);
+
+	/* TODO: Provide mechanism for loading certain attributes at
+	   specified sizes, allow to scale or cap, or scale non-zero */
+	if (ATT_MASK == att) {
+	    atty = ATTY_MASK;
+	}
+	else {
+	    atty = Gs_numtype(filename, &neg);
+	}
+
+#ifdef MAYBE_LATER
+	if (att == ATT_COLOR && atty == ATTY_SHORT) {
+	    atty = (neg ? ATTY_INT : ATTY_SHORT);
+	}
+#endif
+
+	if (att == ATT_COLOR && atty == ATTY_SHORT) {
+	    atty = ATTY_INT;
+	}
+
+	if (0 == gs_malloc_att_buff(gs, att, ATTY_NULL)) {
+	    G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	}
+
+	switch (atty) {
+	case ATTY_MASK:
+	    if (0 == gs_malloc_att_buff(gs, att, ATTY_MASK)) {
+		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	    }
+
+	    ret = Gs_loadmap_as_bitmap(&wind, filename, tbuff->bm);
+	    
+	    break;
+	case ATTY_CHAR:
+	    if (0 == gs_malloc_att_buff(gs, att, ATTY_CHAR)) {
+		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	    }
+
+	    ret = Gs_loadmap_as_char(&wind, filename, tbuff->cb,
+				     tbuff->nm, &has_null);
+
+	    break;
+	case ATTY_SHORT:
+	    if (0 == gs_malloc_att_buff(gs, att, ATTY_SHORT)) {
+		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	    }
+
+	    ret = Gs_loadmap_as_short(&wind, filename, tbuff->sb,
+				      tbuff->nm, &has_null);
+	    break;
+	case ATTY_FLOAT:
+	    if (0 == gs_malloc_att_buff(gs, att, ATTY_FLOAT)) {
+		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	    }
+
+	    ret = Gs_loadmap_as_float(&wind, filename, tbuff->fb,
+				      tbuff->nm, &has_null);
+
+	    break;
+	case ATTY_INT:
+	default:
+	    if (0 == gs_malloc_att_buff(gs, att, ATTY_INT)) {
+		G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+	    }
+
+	    ret = Gs_loadmap_as_int(&wind, filename, tbuff->ib,
+				    tbuff->nm, &has_null);
+	    break;
+
+	}			/* Done with switch */
+
+	if (ret == -1) {
+	    gsds_free_data_buff(gs->att[att].hdata, ATTY_NULL);
+	    return -1;
+	}
+
+	G_debug(4, "  has_null=%d", has_null);
+
+	if (!has_null) {
+	    gsds_free_data_buff(gs->att[att].hdata, ATTY_NULL);
+	}
+	else {
+	    gs_update_curmask(gs);
+	}
+
+    }				/* end if not reuse */
+
+    if (ATT_COLOR == att) {
+#ifdef MAYBE_LATER
+	if (ATTY_INT == atty) {
+	    Gs_pack_colors(filename, tbuff->ib, gs->rows, gs->cols);
+	    gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
+	    gs->att[att].lookup = NULL;
+	}
+	else {
+	    gs_malloc_lookup(gs, att);
+	    Gs_build_lookup(filename, gs->att[att].lookup);
+	}
+#else
+
+	if (ATTY_CHAR == atty) {
+	    if (!gs->att[att].lookup) {
+		/* might already exist if reusing */
+		gs_malloc_lookup(gs, att);
+		Gs_build_256lookup(filename, gs->att[att].lookup);
+	    }
+	}
+	else if (ATTY_FLOAT == atty) {
+	    if (!reuse) {
+		if (0 == gs_malloc_att_buff(gs, att, ATTY_INT)) {
+		    G_fatal_error(_("GS_load_att_map(): Out of memory. Unable to load map"));
+		}
+
+		Gs_pack_colors_float(filename, tbuff->fb, tbuff->ib,
+				     gs->rows, gs->cols);
+		gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
+		gsds_free_data_buff(gs->att[att].hdata, ATTY_FLOAT);
+		gs->att[att].lookup = NULL;
+	    }
+	}
+	else {
+	    if (!reuse) {
+		Gs_pack_colors(filename, tbuff->ib, gs->rows, gs->cols);
+		gsds_set_changed(gs->att[att].hdata, CF_COLOR_PACKED);
+		gs->att[att].lookup = NULL;
+	    }
+	}
+#endif
+    }
+
+    if (ATT_TOPO == att) {
+	gs_init_normbuff(gs);
+	/* S_DIFF: should also check here to see if this surface is a
+	   reference surface for scaled differences, if so update references
+	   to it */
+    }
+
+    if (ret < 0) {
+	G_warning(_("Loading failed"));
+    }
+
+    if (-1 == Gs_update_attrange(gs, att)) {
+	G_warning(_("Error finding range"));
+    }
+
+    return ret;
+}
+
+/*!
+   \brief Draw surface
+
+   \param id surface id
+ */
+void GS_draw_surf(int id)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_draw_surf(): id=%d", id);
+
+    gs = gs_get_surf(id);
+    if (gs) {
+	gsd_shademodel(gs->draw_mode & DM_GOURAUD);
+
+	if (gs->draw_mode & DM_POLY) {
+	    gsd_surf(gs);
+	}
+
+	if (gs->draw_mode & DM_WIRE) {
+	    gsd_wire_surf(gs);
+	}
+
+	/* TODO: write wire/poly draw routines */
+	if (gs->draw_mode & DM_WIRE_POLY) {
+	    gsd_surf(gs);
+	    gsd_wire_surf(gs);
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw surface wire
+
+   Overrides draw_mode for fast display
+
+   \param id surface id
+ */
+void GS_draw_wire(int id)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_draw_wire(): id=%d", id);
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	gsd_wire_surf(gs);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all wires
+
+   Overrides draw_mode for fast display
+ */
+void GS_alldraw_wire(void)
+{
+    geosurf *gs;
+    int i;
+
+    for (i = 0; i < Next_surf; i++) {
+	if ((gs = gs_get_surf(Surf_ID[i]))) {
+	    gsd_wire_surf(gs);
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all surfaces
+ */
+void GS_alldraw_surf(void)
+{
+    int i;
+
+    for (i = 0; i < Next_surf; i++) {
+	GS_draw_surf(Surf_ID[i]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Set Z exag for surface
+
+   \param id surface id
+   \param exag z-exag value
+ */
+void GS_set_exag(int id, float exag)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_exag");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	if (gs->z_exag != exag) {
+	    gs->norm_needupdate = 1;
+	}
+
+	gs->z_exag = exag;
+    }
+
+    return;
+}
+
+/*!
+   \brief Set global z-exag value
+
+   \param exag exag value to be set up
+ */
+void GS_set_global_exag(float exag)
+{
+
+    G_debug(3, "GS_set_global_exag");
+
+    Gv.vert_exag = exag;
+    /* GL_NORMALIZE */
+    /* Only need to update norms gs_norms.c
+     * if exag is used in norm equation which
+     * it is not! If GL_NORMALIZE is disabled
+     * will need to include.
+     gs_setall_norm_needupdate();
+     */
+
+    return;
+}
+
+/*!
+   \brief Get global z-exag value
+
+   \return value
+ */
+float GS_global_exag(void)
+{
+    G_debug(3, "GS_global_exag(): %g", Gv.vert_exag);
+
+    return (Gv.vert_exag);
+}
+
+/*!
+   \brief Set wire color
+
+   \todo error-handling
+
+   \param id surface id
+   \param colr color value
+ */
+void GS_set_wire_color(int id, int colr)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_wire_color");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	gs->wire_color = colr;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get wire color
+
+   \param id surface id
+   \param[out] colr color value
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GS_get_wire_color(int id, int *colr)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*colr = gs->wire_color;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set all draw-modes
+
+   \param mode mode id
+
+   \return 0 on success
+   \return -1 on error
+ */
+int GS_setall_drawmode(int mode)
+{
+    int i;
+
+    for (i = 0; i < Next_surf; i++) {
+	if (0 != GS_set_drawmode(Surf_ID[i], mode)) {
+	    return (-1);
+	}
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Set draw mode
+
+   \param id surface id
+   \param mode mode type(s)
+
+   \return 0 on success
+   \return -1 on error (invalid surface id)
+ */
+int GS_set_drawmode(int id, int mode)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_drawmode(): id=%d mode=%d", id, mode);
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	gs->draw_mode = mode;
+
+	return (0);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get draw mode
+
+   \param id surface id
+   \param[out] mode mode id
+
+   \return 1 on success
+   \return -1 on error (invalid surface id)
+ */
+int GS_get_drawmode(int id, int *mode)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*mode = gs->draw_mode;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set no-zero ?
+
+   \param id surface id
+   \param att attribute id
+   \param mode mode id
+ */
+void GS_set_nozero(int id, int att, int mode)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_nozero");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	if (att == ATT_TOPO) {
+	    gs->nz_topo = mode;
+	    gs->mask_needupdate = 1;
+	}
+
+	if (att == ATT_COLOR) {
+	    gs->nz_color = mode;
+	    gs->mask_needupdate = 1;
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Get no-zero ?
+
+   \param id surface id
+   \param att attribute id
+   \param[out] mode mode id
+
+   \return -1 on error (invalid surface id)
+   \return 1 on success
+ */
+int GS_get_nozero(int id, int att, int *mode)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_nozero");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	if (att == ATT_TOPO) {
+	    *mode = gs->nz_topo;
+	}
+	else if (att == ATT_COLOR) {
+	    *mode = gs->nz_color;
+	}
+	else {
+	    return (-1);
+	}
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set all draw resolutions
+
+   \param xres,yres x/y resolution value
+   \param xwire,ywire x/y wire value
+
+   \return 0 on success
+   \return -1 on error
+ */
+int GS_setall_drawres(int xres, int yres, int xwire, int ywire)
+{
+    int i;
+
+    for (i = 0; i < Next_surf; i++) {
+	if (0 != GS_set_drawres(Surf_ID[i], xres, yres, xwire, ywire)) {
+	    return (-1);
+	}
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Set draw resolution for surface
+
+   \param id surface id
+   \param xres,yres x/y resolution value
+   \param xwire,ywire x/y wire value
+
+   \return -1 on error
+   \return 0 on success
+ */
+int GS_set_drawres(int id, int xres, int yres, int xwire, int ywire)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_set_drawres() id=%d xyres=%d/%d xywire=%d/%d",
+	    id, xres, yres, xwire, ywire);
+
+    if (xres < 1 || yres < 1 || xwire < 1 || ywire < 1) {
+	return (-1);
+    }
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	if (gs->x_mod != xres || gs->y_mod != yres) {
+	    gs->norm_needupdate = 1;
+	}
+
+	gs->x_mod = xres;
+	gs->y_mod = yres;
+	gs->x_modw = xwire;
+	gs->y_modw = ywire;
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Get draw resolution of surface
+
+   \param id surface id
+   \param[out] xres,yres x/y resolution value
+   \param[out] xwire,ywire x/y wire value
+ */
+void GS_get_drawres(int id, int *xres, int *yres, int *xwire, int *ywire)
+{
+    geosurf *gs;
+
+    G_debug(3, "GS_get_drawres");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*xres = gs->x_mod;
+	*yres = gs->y_mod;
+	*xwire = gs->x_modw;
+	*ywire = gs->y_modw;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get dimension of surface
+
+   \param id surface id
+   \param[out] rows,cols number of rows/cols
+ */
+void GS_get_dims(int id, int *rows, int *cols)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*rows = gs->rows;
+	*cols = gs->cols;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get exag-value guess
+
+   Use no_zero range because if zero IS data, then range won't be that
+   much off (it's just a GUESS, after all), but if zero is NO data, could
+   drastically affect guess
+
+   \param id surface id
+   \param[out] exag exag value
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GS_get_exag_guess(int id, float *exag)
+{
+    geosurf *gs;
+    float guess;
+
+    gs = gs_get_surf(id);
+    guess = 1.0;
+
+    /* if gs is type const return guess = 1.0 */
+    if (CONST_ATT == gs_get_att_src(gs, ATT_TOPO)) {
+	*exag = guess;
+	return (1);
+    }
+
+    if (gs) {
+	if (gs->zrange_nz == 0.0) {
+	    *exag = 0.0;
+
+	    return (1);
+	}
+
+	G_debug(3, "GS_get_exag_guess(): %f %f", gs->zrange_nz, Longdim);
+
+	while (gs->zrange_nz * guess / Longdim >= .25) {
+	    guess *= .1;
+
+	    G_debug(3, "GS_get_exag_guess(): %f", guess);
+	}
+
+	while (gs->zrange_nz * guess / Longdim < .025) {
+	    guess *= 10.;
+
+	    G_debug(3, "GS_get_exag_guess(): %f", guess);
+	}
+
+	*exag = guess;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get Z extents for all loaded surfaces
+
+   Treating zeros as "no data"
+
+   \param[out] min min value
+   \param[out] max max value
+ */
+void GS_get_zrange_nz(float *min, float *max)
+{
+    int i, first = 1;
+    geosurf *gs;
+
+    for (i = 0; i < Next_surf; i++) {
+	if ((gs = gs_get_surf(Surf_ID[i]))) {
+	    if (first) {
+		first = 0;
+		*min = gs->zmin_nz;
+		*max = gs->zmax_nz;
+	    }
+
+	    if (gs->zmin_nz < *min) {
+		*min = gs->zmin_nz;
+	    }
+
+	    if (gs->zmax_nz > *max) {
+		*max = gs->zmax_nz;
+	    }
+	}
+    }
+
+    G_debug(3, "GS_get_zrange_nz(): min=%g max=%g", *min, *max);
+
+    return;
+}
+
+/*!
+   \brief Set translation (surface position)
+
+   \param id surface id
+   \param xtrans,ytrans,ztrans translation values
+ */
+void GS_set_trans(int id, float xtrans, float ytrans, float ztrans)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	gs->x_trans = xtrans;
+	gs->y_trans = ytrans;
+	gs->z_trans = ztrans;
+    }
+
+    G_debug(3, "GS_set_trans(): id=%d, x=%f, y=%f, z=%f",
+	    id, xtrans, ytrans, ztrans);
+
+    return;
+}
+
+/*!
+   \brief Get translation values (surface position)
+
+   \param id surface id
+   \param[out] xtrans,ytrans,ztrans trans values
+ */
+void GS_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
+{
+    geosurf *gs;
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	*xtrans = gs->x_trans;
+	*ytrans = gs->y_trans;
+	*ztrans = gs->z_trans;
+    }
+
+    G_debug(3, "GS_get_trans: id=%d, x=%f, y=%f, z=%f",
+	    id, *xtrans, *ytrans, *ztrans);
+
+    return;
+}
+
+
+/*!
+   \brief Get default draw color
+
+   \return color value
+ */
+unsigned int GS_default_draw_color(void)
+{
+
+    G_debug(3, "GS_default_draw_color");
+
+    return ((unsigned int)Gd.bgcol);
+}
+
+/*!
+   \brief Get background color
+
+   \return color value
+ */
+unsigned int GS_background_color(void)
+{
+    return ((unsigned int)Gd.bgcol);
+}
+
+/*!
+   \brief Sets which buffer to draw to
+
+   \param where GSD_BOTH, GSD_FRONT, GSD_BACK
+ */
+void GS_set_draw(int where)
+{
+    Buffermode = where;
+
+    switch (where) {
+    case GSD_BOTH:
+	gsd_bothbuffers();
+
+	break;
+    case GSD_FRONT:
+	gsd_frontbuffer();
+
+	break;
+    case GSD_BACK:
+    default:
+	gsd_backbuffer();
+
+	break;
+    }
+
+    return;
+}
+
+/*
+   \brief Ready to draw
+ */
+void GS_ready_draw(void)
+{
+
+    G_debug(3, "GS_ready_draw");
+
+    gsd_set_view(&Gv, &Gd);
+
+    return;
+}
+
+/*!
+   \brief Draw done, swap buffers
+ */
+void GS_done_draw(void)
+{
+
+    G_debug(3, "GS_done_draw");
+
+    if (GSD_BACK == Buffermode) {
+	gsd_swapbuffers();
+    }
+
+    gsd_flush();
+
+    return;
+}
+
+/*!
+   \brief Set focus
+
+   \param realto real coordinates to
+ */
+void GS_set_focus(float *realto)
+{
+
+    G_debug(3, "GS_set_focus(): %f,%f,%f", realto[0], realto[1], realto[2]);
+
+    Gv.infocus = 1;
+    GS_v3eq(Gv.real_to, realto);
+
+    gsd_set_view(&Gv, &Gd);
+
+    return;
+}
+
+/*!
+   \brief Set real focus
+
+   \param realto real coordinates to
+ */
+void GS_set_focus_real(float *realto)
+{
+
+    G_get_set_window(&wind);
+    realto[X] = realto[X] - wind.west - (wind.ew_res / 2.);
+    realto[Y] = realto[Y] - wind.south - (wind.ns_res / 2.);
+
+    Gv.infocus = 1;
+    GS_v3eq(Gv.real_to, realto);
+
+    gsd_set_view(&Gv, &Gd);
+
+    return;
+}
+
+
+/*!
+   \brief Get focus
+
+   OK to call with NULL argument if just want to check state
+
+   \param realto real coordinates to
+
+   \return ?
+ */
+int GS_get_focus(float *realto)
+{
+
+    G_debug(3, "GS_get_focus");
+
+    if (Gv.infocus) {
+	if (realto) {
+	    GS_v3eq(realto, Gv.real_to);
+	}
+    }
+
+    return (Gv.infocus);
+}
+
+/*!
+   \brief Set focus to map center
+
+   \param id surface id
+ */
+void GS_set_focus_center_map(int id)
+{
+    float center[3];
+    geosurf *gs;
+
+    G_debug(3, "GS_set_focus_center_map");
+
+    gs = gs_get_surf(id);
+
+    if (gs) {
+	center[X] = (gs->xmax - gs->xmin) / 2.;
+	center[Y] = (gs->ymax - gs->ymin) / 2.;
+	center[Z] = (gs->zmax_nz + gs->zmin_nz) / 2.;
+
+	/* not yet working
+	   buff = gs_get_att_typbuff(gs, ATT_TOPO, 0);
+	   offset = gs->rows*gs->cols/2 + gs->cols/2;
+	   if (buff)
+	   {
+	   if (GET_MAPATT(buff, offset, tmp))
+	   {
+	   center[Z] = tmp;
+	   }
+	   }
+	 */
+
+	GS_set_focus(center);
+    }
+}
+
+/*!
+   \brief Move viewpoint 
+
+   \param pt 'from' model coordinates
+ */
+void GS_moveto(float *pt)
+{
+    float ft[3];
+
+    G_debug(3, "GS_moveto(): %f,%f,%f", pt[0], pt[1], pt[2]);
+
+    if (Gv.infocus) {
+	GS_v3eq(Gv.from_to[FROM], pt);
+	/*
+	   GS_v3eq(Gv.from_to[TO], Gv.real_to);
+	 */
+	GS_v3normalize(Gv.from_to[FROM], Gv.from_to[TO]);
+	/* update inclination, look_dir if we're keeping these */
+    }
+    else {
+	GS_v3eq(ft, Gv.from_to[TO]);
+	GS_v3sub(ft, Gv.from_to[FROM]);
+	GS_v3eq(Gv.from_to[FROM], pt);
+	GS_v3eq(Gv.from_to[TO], pt);
+	GS_v3add(Gv.from_to[TO], ft);
+    }
+
+    return;
+}
+
+/*!
+   \brief Move position to (real)
+
+   \param pt point real coordinates
+ */
+void GS_moveto_real(float *pt)
+{
+    gsd_real2model(pt);
+    GS_moveto(pt);
+
+    return;
+}
+
+/*!
+   \brief Get z-extent for a single surface
+
+   \param id surface id
+   \param[out] min min z-value
+   \param[out] max max z-value
+   \param[out] mid middle z-value
+
+   \return -1 on error (invalid surface id)
+   \return ?
+ */
+int GS_get_zextents(int id, float *min, float *max, float *mid)
+{
+    geosurf *gs;
+
+    if (NULL == (gs = gs_get_surf(id))) {
+	return (-1);
+    }
+
+    G_debug(3, "GS_get_zextents(): id=%d", id);
+
+    return (gs_get_zextents(gs, min, max, mid));
+}
+
+/*!
+   \brief Get z-extent for all loaded surfaces
+
+   \param[out] min min z-value
+   \param[out] max max z-value
+   \param doexag use z-exaggeration
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GS_get_zrange(float *min, float *max, int doexag)
+{
+    int ret_surf, ret_vol;
+    float surf_min, surf_max;
+    float vol_min, vol_max;
+
+    ret_surf = gs_get_zrange(&surf_min, &surf_max);
+    ret_vol = gvl_get_zrange(&vol_min, &vol_max);
+
+    if (ret_surf > 0 && ret_vol > 0) {
+	*min = (surf_min < vol_min) ? surf_min : vol_min;
+	*max = (surf_max < vol_max) ? surf_max : vol_max;
+    }
+    else if (ret_surf > 0) {
+	*min = surf_min;
+	*max = surf_max;
+    }
+    else if (ret_vol > 0) {
+	*min = vol_min;
+	*max = vol_max;
+    }
+
+    if (doexag) {
+	*min *= Gv.vert_exag;
+	*max *= Gv.vert_exag;
+    }
+
+    G_debug(3, "GS_get_zrange(): min=%g max=%g", *min, *max);
+    return ((ret_surf > 0 || ret_vol > 0) ? (1) : (-1));
+}
+
+/*!
+   \brief Get viewpoint 'from' position
+
+   \param[out] fr from model coordinates
+ */
+void GS_get_from(float *fr)
+{
+    GS_v3eq(fr, Gv.from_to[FROM]);
+
+    G_debug(3, "GS_get_from(): %f,%f,%f", fr[0], fr[1], fr[2]);
+
+    return;
+}
+
+/*!
+   \brief Get viewpoint 'from' real coordinates
+
+   \param[out] fr 'from' real coordinates
+ */
+void GS_get_from_real(float *fr)
+{
+    GS_v3eq(fr, Gv.from_to[FROM]);
+    gsd_model2real(fr);
+
+    return;
+}
+
+/*!
+   \brief Get 'to' real coordinates
+
+   \param[out] to 'to' real coordinates
+ */
+void GS_get_to_real(float *to)
+{
+    float realto[3];
+
+    G_get_set_window(&wind);
+    GS_get_focus(realto);
+    to[X] = realto[X] + wind.west + (wind.ew_res / 2.);
+    to[Y] = realto[Y] + wind.south + (wind.ns_res / 2.);
+    to[Z] = realto[Z];
+
+    return;
+}
+
+
+/*!
+   \brief Get zoom setup
+
+   \param[out] a,b,c,d current viewport settings
+   \param[out] maxx,maxy max viewport size
+ */
+void GS_zoom_setup(int *a, int *b, int *c, int *d, int *maxx, int *maxy)
+{
+    GLint tmp[4];
+    GLint num[2];
+
+    gsd_getViewport(tmp, num);
+    *a = tmp[0];
+    *b = tmp[1];
+    *c = tmp[2];
+    *d = tmp[3];
+    *maxx = num[0];
+    *maxy = num[1];
+
+    return;
+}
+
+/*!
+   \brief Get 'to' model coordinates
+
+   \todo need set_to? - just use viewdir?
+
+   \param[out] to 'to' model coordinates
+ */
+void GS_get_to(float *to)
+{
+    G_debug(3, "GS_get_to");
+
+    GS_v3eq(to, Gv.from_to[TO]);
+
+    return;
+}
+
+/*!
+   \brief Get viewdir
+
+   \param[out] dir viewdir value
+ */
+void GS_get_viewdir(float *dir)
+{
+    GS_v3dir(Gv.from_to[FROM], Gv.from_to[TO], dir);
+
+    return;
+}
+
+/*!
+   \brief Set viewdir
+
+   Automatically turns off focus
+
+   \param dir viewdir value
+ */
+void GS_set_viewdir(float *dir)
+{
+    float tmp[3];
+
+    GS_v3eq(tmp, dir);
+    GS_v3norm(tmp);
+    GS_v3eq(Gv.from_to[TO], Gv.from_to[FROM]);
+    GS_v3add(Gv.from_to[TO], tmp);
+
+    GS_set_nofocus();
+    gsd_set_view(&Gv, &Gd);
+
+    return;
+}
+
+/*!
+   \brief Set field of view
+
+   \param fov fov value
+ */
+void GS_set_fov(int fov)
+{
+    Gv.fov = fov;
+
+    return;
+}
+
+/*!
+   \brief Get fied of view
+
+   \return field of view, in 10ths of degrees
+ */
+int GS_get_fov(void)
+{
+    return (Gv.fov);
+}
+
+/*!
+   \brief Get twist value
+
+   10ths of degrees off twelve o'clock
+ */
+int GS_get_twist(void)
+{
+    return (Gv.twist);
+}
+
+/*!
+   \brief Set viewpoint twist value
+
+   10ths of degrees off twelve o'clock
+
+   \param t tenths of degrees clockwise from 12:00.
+ */
+void GS_set_twist(int t)
+{
+    Gv.twist = t;
+
+    return;
+}
+
+/*!
+   \brief Set rotation params
+ */
+void GS_set_rotation(double angle, double x, double y, double z)
+{
+    Gv.rotate.rot_angle = angle;
+    Gv.rotate.rot_axes[0] = x;
+    Gv.rotate.rot_axes[1] = y;
+    Gv.rotate.rot_axes[2] = z;
+    Gv.rotate.do_rot = 1;
+
+    return;
+}
+
+/*!
+   \brief Stop scene rotation
+ */
+void GS_unset_rotation(void)
+{
+    Gv.rotate.do_rot = 0;
+}
+
+/*!
+   \brief Reset scene rotation
+ */
+void GS_init_rotation(void)
+{
+    int i;
+
+    for (i = 0; i < 16; i++) {
+	if (i == 0 || i == 5 || i == 10 || i == 15)
+	    Gv.rotate.rotMatrix[i] = 1.0;
+	else
+	    Gv.rotate.rotMatrix[i] = 0.0;
+    }
+    Gv.rotate.rot_angle = 0.0;
+    Gv.rotate.rot_axes[0] = 0.0;
+    Gv.rotate.rot_axes[1] = 0.0;
+    Gv.rotate.rot_axes[2] = 0.0;
+    Gv.rotate.do_rot = 0;
+    
+}
+/*!
+ * \brief Get rotation matrix
+ */ 
+void GS_get_rotation_matrix(double *matrix)
+{
+    int i;
+
+    for (i = 0; i < 16; i++) {
+	matrix[i] = Gv.rotate.rotMatrix[i];
+    }
+}
+
+/*!
+ * \brief Set rotation matrix
+ */ 
+void GS_set_rotation_matrix(double *matrix)
+{
+    int i;
+
+    for (i = 0; i < 16; i++) {
+	Gv.rotate.rotMatrix[i] = matrix[i];
+    }
+}
+
+/*!
+   \brief Unset focus
+ */
+void GS_set_nofocus(void)
+{
+    G_debug(3, "GS_set_nofocus");
+
+    Gv.infocus = 0;
+
+    return;
+}
+
+/*!
+   \brief Set focus
+
+   Make sure that the center of view is set
+ */
+void GS_set_infocus(void)
+{
+    G_debug(3, "GS_set_infocus");
+
+    Gv.infocus = 1;
+
+    return;
+}
+
+/*!
+   \brief Set viewport
+
+   \param left,right,bottom,top viewport extent values
+ */
+void GS_set_viewport(int left, int right, int bottom, int top)
+{
+    G_debug(3, "GS_set_viewport(): left=%d, right=%d, "
+	    "bottom=%d, top=%d", left, right, bottom, top);
+
+    gsd_viewport(left, right, bottom, top);
+
+    return;
+}
+
+/*!
+   \brief Send screen coords sx and sy, lib traces through surfaces; sets
+   new center to point of nearest intersection.
+
+   If no intersection, uses line of sight with length of current view
+   ray (eye to center) to set new center.
+
+   Reset center of view to screen coordinates sx, sy.
+
+   \param sx,sy screen coordinates
+
+   \return 1 on success
+   \return 0 on error (invalid surface id)
+ */
+int GS_look_here(int sx, int sy)
+{
+    float x, y, z, len, los[2][3];
+    Point3 realto, dir;
+    int id;
+    geosurf *gs;
+
+    if (GS_get_selected_point_on_surface(sx, sy, &id, &x, &y, &z)) {
+	gs = gs_get_surf(id);
+	if (gs) {
+	    realto[X] = x - gs->ox + gs->x_trans;
+	    realto[Y] = y - gs->oy + gs->y_trans;
+	    realto[Z] = z + gs->z_trans;
+	    GS_set_focus(realto);
+
+	    return (1);
+	}
+    }
+    else {
+	if (gsd_get_los(los, (short)sx, (short)sy)) {
+	    len = GS_distance(Gv.from_to[FROM], Gv.real_to);
+	    GS_v3dir(los[FROM], los[TO], dir);
+	    GS_v3mult(dir, len);
+	    realto[X] = Gv.from_to[FROM][X] + dir[X];
+	    realto[Y] = Gv.from_to[FROM][Y] + dir[Y];
+	    realto[Z] = Gv.from_to[FROM][Z] + dir[Z];
+	    GS_set_focus(realto);
+
+	    return (1);
+	}
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Get selected point of surface
+
+   Given screen coordinates sx and sy, find closest intersection of
+   view ray with surfaces and return coordinates of intersection in x, y,
+   z, and identifier of surface in id.
+
+   \param sx,sy screen coordinates
+   \param[out] id surface id
+   \param[out] x,y,z point on surface (model coordinates?)
+
+   \returns 0 if no intersections found
+   \return number of intersections
+ */
+int GS_get_selected_point_on_surface(int sx, int sy, int *id, float *x,
+				     float *y, float *z)
+{
+    float los[2][3], find_dist[MAX_SURFS], closest;
+    Point3 point, tmp, finds[MAX_SURFS];
+    int surfs[MAX_SURFS], i, iclose, numhits = 0;
+    geosurf *gs;
+
+    /* returns surface-world coords */
+    gsd_get_los(los, (short)sx, (short)sy);
+
+    if (!gs_setlos_enterdata(los)) {
+	G_debug(3, "gs_setlos_enterdata(los): returns false");
+	return (0);
+    }
+
+    for (i = 0; i < Next_surf; i++) {
+	G_debug(3, "id=%d", i);
+
+	gs = gs_get_surf(Surf_ID[i]);
+
+	/* los_intersect expects surf-world coords (xy transl, no scaling) */
+
+#if NVIZ_HACK
+	if (gs_los_intersect1(Surf_ID[i], los, point)) {
+#else
+	if (gs_los_intersect(Surf_ID[i], los, point)) {
+#endif
+	    if (!gs_point_is_masked(gs, point)) {
+		GS_v3eq(tmp, point);
+		tmp[X] += gs->x_trans;
+		tmp[Y] += gs->y_trans;
+		tmp[Z] += gs->z_trans;
+		find_dist[numhits] = GS_distance(los[FROM], tmp);
+		gsd_surf2real(gs, point);
+		GS_v3eq(finds[numhits], point);
+		surfs[numhits] = Surf_ID[i];
+		numhits++;
+	    }
+	}
+    }
+
+    for (i = iclose = 0; i < numhits; i++) {
+	closest = find_dist[iclose];
+
+	if (find_dist[i] < closest) {
+	    iclose = i;
+	}
+    }
+
+    if (numhits) {
+	*x = finds[iclose][X];
+	*y = finds[iclose][Y];
+	*z = finds[iclose][Z];
+	*id = surfs[iclose];
+    }
+
+    G_debug(3, "NumHits %d, next %d", numhits, Next_surf);
+
+    return (numhits);
+}
+
+/*!
+   \brief Set cplace rotation
+
+   \param num cplace id
+   \param dx,dy,dz rotation values
+ */
+void GS_set_cplane_rot(int num, float dx, float dy, float dz)
+{
+    gsd_cplane_setrot(num, dx, dy, dz);
+
+    return;
+}
+
+/*!
+   \brief Set cplace trans
+
+   \param num cplace id
+   \param dx,dy,dz rotation values
+ */
+void GS_set_cplane_trans(int num, float dx, float dy, float dz)
+{
+    gsd_cplane_settrans(num, dx, dy, dz);
+
+    return;
+}
+
+
+/*!
+   \brief Draw cplace
+
+   \param num cplace id
+ */
+void GS_draw_cplane(int num)
+{
+    geosurf *gsurfs[MAX_SURFS];
+    int nsurfs;
+
+    nsurfs = gs_num_surfaces();
+    if (2 == nsurfs) {
+	/* testing */
+	gs_getall_surfaces(gsurfs);
+	gsd_draw_cplane_fence(gsurfs[0], gsurfs[1], num);
+    }
+    else {
+	gsd_draw_cplane(num);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw cplace fence ?
+
+   \param hs1,hs2
+   \param num cplane id
+
+   \return 0 on error
+   \return 1 on success
+ */
+int GS_draw_cplane_fence(int hs1, int hs2, int num)
+{
+    geosurf *gs1, *gs2;
+
+    if (NULL == (gs1 = gs_get_surf(hs1))) {
+	return (0);
+    }
+
+    if (NULL == (gs2 = gs_get_surf(hs2))) {
+	return (0);
+    }
+
+    gsd_draw_cplane_fence(gs1, gs2, num);
+
+    return (1);
+}
+
+/*!
+   \brief Draw all cplace fences ?
+ */
+void GS_alldraw_cplane_fences(void)
+{
+    int onstate[MAX_CPLANES], i;
+
+    gsd_get_cplanes_state(onstate);
+
+    for (i = 0; i < MAX_CPLANES; i++) {
+	if (onstate[i]) {
+	    GS_draw_cplane_fence(Surf_ID[0], Surf_ID[1], i);
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Set cplace
+
+   \param num cplane id
+ */
+void GS_set_cplane(int num)
+{
+    gsd_cplane_on(num);
+
+    return;
+}
+
+/*!
+   \brief Unset clip place (turn off)
+
+   \param num cplane id
+ */
+void GS_unset_cplane(int num)
+{
+    gsd_cplane_off(num);
+
+    return;
+}
+
+/*!
+   \brief Get axis scale
+
+   \param sx,sy,sz x/y/z scale values
+   \param doexag use vertical exaggeration
+ */
+void GS_get_scale(float *sx, float *sy, float *sz, int doexag)
+{
+    float zexag;
+
+    zexag = doexag ? Gv.vert_exag : 1.;
+    *sx = *sy = Gv.scale;
+    *sz = Gv.scale * zexag;
+
+    return;
+}
+
+/*!
+   \brief Set fence color
+
+   \param mode mode id
+ */
+void GS_set_fencecolor(int mode)
+{
+    gsd_setfc(mode);
+
+    return;
+}
+
+/*!
+   \brief Get fence color
+
+   \return color value
+ */
+int GS_get_fencecolor(void)
+{
+    return gsd_getfc();
+}
+
+/*!
+   \brief Measure distance "as the ball rolls" between two points on
+   surface
+
+   \param hs surface id
+   \param x1,y1,x2,y2 two points on surface
+   \param[out] dist measured distance
+   \param use_exag use exag. surface
+
+   \return 0 on error or if one or more points is not in region
+   \return distance following terrain
+ */
+int GS_get_distance_alongsurf(int hs, float x1, float y1, float x2, float y2,
+			      float *dist, int use_exag)
+{
+    geosurf *gs;
+    float p1[2], p2[2];
+    
+    gs = gs_get_surf(hs);
+    if (gs == NULL) {
+	return 0;
+    }
+    
+    p1[X] = x1;
+    p1[Y] = y1;
+    p2[X] = x2;
+    p2[Y] = y2;
+    gsd_real2surf(gs, p1);
+    gsd_real2surf(gs, p2);
+
+    G_debug(3, "GS_get_distance_alongsurf(): hs=%d p1=%f,%f p2=%f,%f",
+	    hs, x1, y1, x2, y2);
+    return gs_distance_onsurf(gs, p1, p2, dist, use_exag);
+}
+
+/*!
+   \brief Save 3d view
+
+   \param vname view file name
+   \param surfid surface id
+
+   \return ?
+ */
+int GS_save_3dview(const char *vname, int surfid)
+{
+    return (Gs_save_3dview(vname, &Gv, &Gd, &wind, gs_get_surf(surfid)));
+}
+
+/*!
+   \brief Load 3d view
+
+   \param vname view file name
+   \param surfid surface id
+
+   \return ?
+ */
+int GS_load_3dview(const char *vname, int surfid)
+{
+
+    return (Gs_load_3dview(vname, &Gv, &Gd, &wind, gs_get_surf(surfid)));
+
+    /* what to do about lights - I guess, delete all &
+       create any that exist in 3dview file */
+}
+
+/************************************************************************
+* Following routines use Graphics Library
+************************************************************************/
+
+/*!
+   \brief Init viewpoint
+
+   \todo allow to set center?
+ */
+void GS_init_view(void)
+{
+    static int first = 1;
+
+    G_debug(3, "GS_init_view");
+
+    if (first) {
+	first = 0;
+	glMatrixMode(GL_MODELVIEW);
+
+	/* OGLXXX doublebuffer: use GLX_DOUBLEBUFFER in attriblist */
+	/* glxChooseVisual(*dpy, screen, *attriblist); */
+	/* OGLXXX
+	 * ZMIN not needed -- always 0.
+	 * ZMAX not needed -- always 1.
+	 * getgdesc other posiblilties:
+	 *      glxGetConfig();
+	 *      glxGetCurrentContext();
+	 *      glxGetCurrentDrawable();
+	 * GLint gdtmp;
+	 * getgdesc other posiblilties:
+	 *      glxGetConfig();
+	 *      glxGetCurrentContext();
+	 *      glxGetCurrentDrawable();
+	 * GLint gdtmp;
+	 * glDepthRange params must be scaled to [0, 1]
+	 */
+	glDepthRange(0.0, 1.0);
+	glEnable(GL_DEPTH_TEST);
+	glDepthFunc(GL_LEQUAL);
+	/* } */
+
+	/* replace these with something meaningful */
+	Gv.fov = 450;
+	Gv.twist = 0;
+
+	GS_init_rotation();
+
+	Gv.from_to[FROM][X] = Gv.from_to[FROM][Y] =
+	    Gv.from_to[FROM][Z] = GS_UNIT_SIZE / 2.;
+
+	Gv.from_to[TO][X] = GS_UNIT_SIZE / 2.;
+	Gv.from_to[TO][Y] = GS_UNIT_SIZE / 2.;
+	Gv.from_to[TO][Z] = 0.;
+	Gv.from_to[TO][W] = Gv.from_to[FROM][W] = 1.;
+
+	Gv.real_to[W] = 1.;
+	Gv.vert_exag = 1.;
+
+	GS_v3eq(Gv.real_to, Gv.from_to[TO]);
+	GS_v3normalize(Gv.from_to[FROM], Gv.from_to[TO]);
+
+	/*
+	   Gd.nearclip = 50;
+	   Gd.farclip = 10000.;
+	 */
+	Gd.nearclip = 10.;
+	Gd.farclip = 10000.;
+	Gd.aspect = (float)GS_get_aspect();
+
+	GS_set_focus(Gv.real_to);
+    }
+
+    return;
+}
+
+/*!
+   \brief Clear view
+
+   \param col color value
+ */
+void GS_clear(int col)
+{
+    G_debug(3, "GS_clear");
+
+    col = col | 0xFF000000;
+
+    /* OGLXXX
+     * change glClearDepth parameter to be in [0, 1]
+     * ZMAX not needed -- always 1.
+     * getgdesc other posiblilties:
+     *      glxGetConfig();
+     *      glxGetCurrentContext();
+     *      glxGetCurrentDrawable();
+     * GLint gdtmp;
+     */
+    glClearDepth(1.0);
+    glClearColor(((float)((col) & 0xff)) / 255.,
+		 (float)((col) >> 8 & 0xff) / 255.,
+		 (float)((col) >> 16 & 0xff) / 255.,
+		 (float)((col) >> 24 & 0xff) / 255.);
+    glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT);
+
+    Gd.bgcol = col;
+    Modelshowing = 0;
+    gsd_flush();
+
+    return;
+}
+
+/*!
+   \brief Get aspect value
+
+   \return aspect value
+ */
+double GS_get_aspect(void)
+{
+    int left, right, bottom, top;
+    GLint tmp[4];
+
+    /* OGLXXX
+     * get GL_VIEWPORT:
+     * You can probably do better than this.
+     */
+    glGetIntegerv(GL_VIEWPORT, tmp);
+    left = tmp[0];
+    right = tmp[0] + tmp[2] - 1;
+    bottom = tmp[1];
+    top = tmp[1] + tmp[3] - 1;
+
+    G_debug(3, "GS_get_aspect(): left=%d, right=%d, top=%d, bottom=%d",
+	    left, right, top, bottom);
+
+    return ((double)(right - left) / (top - bottom));
+}
+
+/*!
+   \brief Check for transparency
+
+   Disabled.
+
+   \return 1
+ */
+int GS_has_transparency(void)
+{
+    /* OGLXXX
+     * getgdesc other posiblilties:
+     *      glxGetConfig();
+     *      glxGetCurrentContext();
+     *      glxGetCurrentDrawable();
+     * GLint gdtmp;
+     * blending is ALWAYS supported.
+     * This function returns whether it is enabled.
+     * return((glGetIntegerv(GL_BLEND, &gdtmp), gdtmp));
+     */
+
+    return (1);
+}

Copied: grass/trunk/lib/ogsf/gs3.c (from rev 62429, grass/trunk/lib/ogsf/Gs3.c)
===================================================================
--- grass/trunk/lib/ogsf/gs3.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gs3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,1198 @@
+/*!
+   \file Gs3.c
+
+   \brief OGSF library - loading surfaces (lower level functions)
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown USACERL, GMSL/University of Illinois (January 1993)
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <stdlib.h>
+#include <string.h>
+
+#include <grass/gis.h>
+#include <grass/raster.h>
+#include <grass/glocale.h>
+#include <grass/bitmap.h>
+
+#include <grass/ogsf.h>
+/* for geoview & geodisplay in 3dview stuff */
+#include "gsget.h"
+/* for update_attrange - might be able to move this func now */
+
+/*!
+   \brief Used in the function Gs_update_attrange()
+ */
+#define INIT_MINMAX(p, nm, size, min, max, found) \
+	found = 0; \
+	p+=(size-1); \
+	while (size--) \
+	{ \
+	    if (!BM_GET_BYOFFSET(nm, size)) \
+	    { \
+		min = max = *p; \
+		found = 1; \
+		break; \
+	    } \
+	    p--; \
+	}
+
+/*!
+   \brief Used in the function Gs_update_attrange()
+ */
+#define SET_MINMAX(p, nm, size, min, max) \
+	p+=(size-1); \
+	while(size--) \
+	{ \
+	    if (!BM_GET_BYOFFSET(nm, size)) \
+	    { \
+		if (*p < min) \
+		{ \
+		    min = *p; \
+		} \
+		else if (*p > max) \
+		{ \
+		    max = *p; \
+		}  \
+	    } \
+	    p--; \
+	}
+
+typedef int FILEDESC;
+
+#define NO_DATA_COL 0xffffff
+
+/*!
+   \brief Calculates distance in METERS between two points in current projection (2D)
+
+   Uses G_distance().
+
+   \param from 'from' point (X, Y)
+   \param to 'to' point (X, Y)
+
+   \return distance
+ */
+double Gs_distance(double *from, double *to)
+{
+    static int first = 1;
+
+    if (first) {
+	first = 0;
+	G_begin_distance_calculations();
+    }
+
+    return G_distance(from[0], from[1], to[0], to[1]);
+}
+
+/*!
+   \brief Load raster map as floating point map
+
+   Calling function must have already allocated space in buff for
+   wind->rows * wind->cols floats.
+
+   This routine simply loads the map into a 2d array by repetitve calls
+   to get_f_raster_row.
+
+   \param wind current window
+   \param map_name raster map name
+   \param[out] buff data buffer
+   \param[out] nullmap null map buffer
+   \param[out] has_null indicates if raster map contains null-data
+
+   \return 1 on success
+   \return 0 on failure
+ */
+int Gs_loadmap_as_float(struct Cell_head *wind, const char *map_name,
+			float *buff, struct BM *nullmap, int *has_null)
+{
+    FILEDESC cellfile;
+    const char *map_set;
+    int offset, row, col;
+
+    G_debug(3, "Gs_loadmap_as_float(): name=%s", map_name);
+
+    map_set = G_find_raster2(map_name, "");
+    if (!map_set) {
+	G_warning(_("Raster map <%s> not found"), map_name);
+	return 0;
+    }
+    *has_null = 0;
+
+    cellfile = Rast_open_old(map_name, map_set);
+
+    G_message(_("Loading raster map <%s>..."),
+	      G_fully_qualified_name(map_name, map_set));
+
+    for (row = 0; row < wind->rows; row++) {
+	offset = row * wind->cols;
+	Rast_get_f_row(cellfile, &(buff[offset]), row);
+
+	G_percent(row, wind->rows, 2);
+
+	for (col = 0; col < wind->cols; col++) {
+	    if (Rast_is_f_null_value(buff + offset + col)) {
+		*has_null = 1;
+		BM_set(nullmap, col, row, 1);
+	    }
+	    /* set nm */
+	}
+    }
+    G_percent(1, 1, 1);
+
+    G_debug(4, "  has_null=%d", *has_null);
+
+    Rast_close(cellfile);
+
+    return (1);
+}
+
+/*!
+   \brief Load raster map as integer map
+
+   Calling function must have already allocated space in buff for
+   wind->rows * wind->cols floats.
+
+   This routine simply loads the map into a 2d array by repetitve calls
+   to get_f_raster_row.
+
+   \todo fn body of Gs_loadmap_as_float()
+
+   \param wind current window
+   \param map_name raster map name
+   \param[out] buff data buffer
+   \param[out] nullmap null map buffer
+   \param[out] has_null indicates if raster map contains null-data
+
+   \return 1 on success
+   \return 0 on failure
+ */
+int Gs_loadmap_as_int(struct Cell_head *wind, const char *map_name, int *buff,
+		      struct BM *nullmap, int *has_null)
+{
+    FILEDESC cellfile;
+    const char *map_set;
+    int offset, row, col;
+
+    G_debug(3, "Gs_loadmap_as_int");
+
+    map_set = G_find_raster2(map_name, "");
+    if (!map_set) {
+	G_warning(_("Raster map <%s> not found"), map_name);
+	return 0;
+    }
+    *has_null = 0;
+
+    cellfile = Rast_open_old(map_name, map_set);
+
+    G_message(_("Loading raster map <%s>..."),
+	      G_fully_qualified_name(map_name, map_set));
+
+    for (row = 0; row < wind->rows; row++) {
+	offset = row * wind->cols;
+	Rast_get_c_row(cellfile, &(buff[offset]), row);
+
+	G_percent(row, wind->rows, 2);
+
+	for (col = 0; col < wind->cols; col++) {
+	    if (Rast_is_f_null_value(buff + offset + col)) {
+		*has_null = 1;
+		BM_set(nullmap, col, row, 1);
+	    }
+
+	    /* set nm */
+	}
+    }
+    G_percent(1, 1, 1);
+    
+    Rast_close(cellfile);
+
+    return (1);
+}
+
+/*!
+   \brief Get map data type
+
+   \param filename raster map name
+   \param negflag
+
+   \return -1 if map is integer and Rast_read_range() fails
+   \return data type (ARRY_*)
+ */
+int Gs_numtype(const char *filename, int *negflag)
+{
+    CELL max = 0, min = 0;
+    struct Range range;
+    const char *mapset;
+    int shortbits, charbits, bitplace;
+    static int max_short, max_char;
+    static int first = 1;
+
+    if (first) {
+	max_short = max_char = 1;
+	shortbits = 8 * sizeof(short);
+
+	for (bitplace = 1; bitplace < shortbits; ++bitplace) {
+	    /*1 bit for sign */
+	    max_short *= 2;
+	}
+
+	max_short -= 1;
+
+	/* NO bits for sign, using unsigned char */
+	charbits = 8 * sizeof(unsigned char);
+
+	for (bitplace = 0; bitplace < charbits; ++bitplace) {
+	    max_char *= 2;
+	}
+
+	max_char -= 1;
+
+	first = 0;
+    }
+
+    mapset = G_find_raster2(filename, "");
+    if (!mapset) {
+	G_warning(_("Raster map <%s> not found"), filename);
+	return -1;
+    }
+
+    if (Rast_map_is_fp(filename, mapset)) {
+	G_debug(3, "Gs_numtype(): fp map detected");
+
+	return (ATTY_FLOAT);
+    }
+
+    if (-1 == Rast_read_range(filename, mapset, &range)) {
+	return (-1);
+    }
+
+    Rast_get_range_min_max(&range, &min, &max);
+    *negflag = (min < 0);
+
+    if (max < max_char && min > 0) {
+	return (ATTY_CHAR);
+    }
+
+    if (max < max_short && min > -max_short) {
+	return (ATTY_SHORT);
+    }
+
+    return (ATTY_INT);
+}
+
+/*!
+   \brief Load raster map as integer map
+
+   Calling function must have already allocated space in buff for
+   wind->rows * wind->cols shorts.  
+
+   This routine simply loads the map into a 2d array by repetitve calls
+   to get_map_row.
+
+   \param wind current window
+   \param map_name raster map name
+   \param[out] buff data buffer
+   \param[out] nullmap null map buffer
+   \param[out] has_null indicates if raster map contains null-data
+
+   \return 1 on success
+   \return -1 on failure,
+   \return -2 if read ok, but 1 or more values were too large (small)
+   to fit into a short (in which case the max (min) short is used)
+ */
+int Gs_loadmap_as_short(struct Cell_head *wind, const char *map_name,
+			short *buff, struct BM *nullmap, int *has_null)
+{
+    FILEDESC cellfile;
+    const char *map_set;
+    int *ti, *tmp_buf;
+    int offset, row, col, val, max_short, overflow, shortsize, bitplace;
+    short *ts;
+
+    G_debug(3, "Gs_loadmap_as_short");
+
+    overflow = 0;
+    shortsize = 8 * sizeof(short);
+
+    /* 1 bit for sign */
+    /* same as 2 << (shortsize-1) */
+    for (max_short = bitplace = 1; bitplace < shortsize; ++bitplace) {
+	max_short *= 2;
+    }
+
+    max_short -= 1;
+
+    map_set = G_find_raster2(map_name, "");
+    if (!map_set) {
+	G_warning(_("Raster map <%s> not found"), map_name);
+	return -1;
+    }
+    *has_null = 0;
+
+    cellfile = Rast_open_old(map_name, map_set);
+
+    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
+    if (!tmp_buf) {
+	return -1;
+    }
+
+    G_message(_("Loading raster map <%s>..."),
+	      G_fully_qualified_name(map_name, map_set));
+
+    for (row = 0; row < wind->rows; row++) {
+	offset = row * wind->cols;
+	Rast_get_c_row(cellfile, tmp_buf, row);
+
+	G_percent(row, wind->rows, 2);
+
+	ts = &(buff[offset]);
+	ti = tmp_buf;
+
+	for (col = 0; col < wind->cols; col++) {
+	    if (Rast_is_c_null_value(&tmp_buf[col])) {
+		*has_null = 1;
+		BM_set(nullmap, col, row, 1);
+	    }
+	    else {
+		val = *ti;
+		if (abs(val) > max_short) {
+		    overflow = 1;
+		    /* assign floor/ceiling value?
+		     */
+		    *ts = (short)(max_short * val / abs(val));
+		}
+		else {
+		    *ts = (short)val;
+		}
+	    }
+
+	    ti++;
+	    ts++;
+	}
+    }
+    G_percent(1, 1, 1);
+    
+    Rast_close(cellfile);
+
+    G_free(tmp_buf);
+
+    return (overflow ? -2 : 1);
+}
+
+/*!
+   \brief Load raster map as integer map
+
+   Calling function must have already allocated space in buff for
+   wind->rows * wind->cols unsigned chars.  
+
+   This routine simply loads the map into a 2d array by repetitve calls
+   to get_map_row.
+
+   Since signs of chars can be tricky, we only load positive chars
+   between 0-255.
+
+   \todo fn body Gs_loadmap_as_float()
+
+   \param wind current window
+   \param map_name raster map name
+   \param[out] buff data buffer
+   \param[out] nullmap null map buffer
+   \param[out] has_null indicates if raster map contains null-data
+
+   \return 1 on success
+   \return -1 on failure
+   \return -2 if read ok, but 1 or more values
+   were too large (small) to fit into an unsigned char.
+   (in which case the max (min) char is used)
+ */
+int Gs_loadmap_as_char(struct Cell_head *wind, const char *map_name,
+		       unsigned char *buff, struct BM *nullmap, int *has_null)
+{
+    FILEDESC cellfile;
+    const char *map_set;
+    int *ti, *tmp_buf;
+    int offset, row, col, val, max_char, overflow, charsize, bitplace;
+    unsigned char *tc;
+
+    G_debug(3, "Gs_loadmap_as_char");
+
+    overflow = 0;
+    charsize = 8 * sizeof(unsigned char);
+
+    /* 0 bits for sign! */
+    max_char = 1;
+
+    for (bitplace = 0; bitplace < charsize; ++bitplace) {
+	max_char *= 2;
+    }
+
+    max_char -= 1;
+
+    map_set = G_find_raster2(map_name, "");
+    if (!map_set) {
+	G_warning(_("Raster map <%s> not found"), map_name);
+	return -1;
+    }
+    *has_null = 0;
+
+    cellfile = Rast_open_old(map_name, map_set);
+
+    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
+    if (!tmp_buf) {
+	return -1;
+    }
+
+    G_message(_("Loading raster map <%s>..."),
+	      G_fully_qualified_name(map_name, map_set));
+
+    for (row = 0; row < wind->rows; row++) {
+	offset = row * wind->cols;
+	Rast_get_c_row(cellfile, tmp_buf, row);
+	tc = (unsigned char *)&(buff[offset]);
+	ti = tmp_buf;
+
+	G_percent(row, wind->rows, 2);
+
+	for (col = 0; col < wind->cols; col++) {
+	    if (Rast_is_c_null_value(&tmp_buf[col])) {
+		*has_null = 1;
+		BM_set(nullmap, col, row, 1);
+	    }
+	    else {
+		val = *ti;
+		if (val > max_char) {
+		    overflow = 1;
+		    *tc = (unsigned char)max_char;
+		}
+		else if (val < 0) {
+		    overflow = 1;
+		    *tc = 0;
+		}
+		else {
+		    *tc = (unsigned char)val;
+		}
+	    }
+
+	    ti++;
+	    tc++;
+	}
+    }
+    G_percent(1, 1, 1);
+    
+    Rast_close(cellfile);
+
+    G_free(tmp_buf);
+
+    return (overflow ? -2 : 1);
+}
+
+/*!
+   \brief Load raster map as integer map
+
+   Calling function must have already allocated space in buff for
+   struct BM of wind->rows & wind->cols.
+
+   This routine simply loads the map into the bitmap by repetitve calls
+   to get_map_row.  Any value other than 0 in the map will set the bitmap.
+   (may want to change later to allow specific value to set)
+
+   Changed to use null.
+
+   \param wind current window
+   \param map_name raster map name
+   \param[out] buff data buffer
+
+   \returns 1 on success
+   \return -1 on failure
+ */
+int Gs_loadmap_as_bitmap(struct Cell_head *wind, const char *map_name,
+			 struct BM *buff)
+{
+    FILEDESC cellfile;
+    const char *map_set;
+    int *tmp_buf;
+    int row, col;
+
+    G_debug(3, "Gs_loadmap_as_bitmap");
+
+    map_set = G_find_raster2(map_name, "");
+    if (!map_set) {
+	G_warning(_("Raster map <%s> not found"), map_name);
+	return -1;
+    }
+
+    cellfile = Rast_open_old(map_name, map_set);
+
+    tmp_buf = (int *)G_malloc(wind->cols * sizeof(int));	/* G_fatal_error */
+    if (!tmp_buf) {
+	return -1;
+    }
+
+    G_message(_("Loading raster map <%s>..."),
+	      G_fully_qualified_name(map_name, map_set));
+
+    for (row = 0; row < wind->rows; row++) {
+	Rast_get_c_row(cellfile, tmp_buf, row);
+
+	for (col = 0; col < wind->cols; col++) {
+	    if (Rast_is_c_null_value(&tmp_buf[col])) {
+		/* no data */
+		BM_set(buff, col, row, 1);
+	    }
+	    else {
+		BM_set(buff, col, row, 0);
+	    }
+	}
+    }
+
+    Rast_close(cellfile);
+
+    G_free(tmp_buf);
+
+    return (1);
+}
+
+/*!
+   \brief Build color table (256)
+
+   Calling function must have already allocated space in buff for range of
+   data (256 for now) - simply calls get_color for each cat in color range
+
+   \param filename raster map name
+   \param[out] buff data buffer
+
+   \return 1 on success
+   \return 0 on failure
+ */
+int Gs_build_256lookup(const char *filename, int *buff)
+{
+    const char *mapset;
+    struct Colors colrules;
+    CELL min, max, cats[256];
+    int i;
+    unsigned char r[256], g[256], b[256], set[256];
+
+    G_debug(3, "building color table");
+
+    mapset = G_find_raster2(filename, "");
+    if (!mapset) {
+	G_warning(_("Raster map <%s> not found"), filename);
+	return 0;
+    }
+
+    Rast_read_colors(filename, mapset, &colrules);
+    Rast_get_c_color_range(&min, &max, &colrules);
+
+    if (min < 0 || max > 255) {
+	G_warning(_("Color table range doesn't match data (mincol=%d, maxcol=%d"),
+		  min, max);
+
+	min = min < 0 ? 0 : min;
+	max = max > 255 ? 255 : max;
+    }
+
+    G_zero(cats, 256 * sizeof(CELL));
+
+    for (i = min; i <= max; i++) {
+	cats[i] = i;
+    }
+
+    Rast_lookup_c_colors(cats, r, g, b, set, 256, &colrules);
+
+    for (i = 0; i < 256; i++) {
+
+	if (set[i]) {
+	    buff[i] =
+		(r[i] & 0xff) | ((g[i] & 0xff) << 8) | ((b[i] & 0xff) << 16);
+	}
+	else {
+	    buff[i] = NO_DATA_COL;
+	}
+    }
+
+    return (1);
+}
+
+/*!
+   \brief Pack color table
+
+   Passed an array of 32 bit ints that is converted from cell values
+   to packed colors (0xbbggrr) 
+
+   \param filename raster map name
+   \param buff
+   \param rows number of rows
+   \param cols number of cols
+ */
+void Gs_pack_colors(const char *filename, int *buff, int rows, int cols)
+{
+    const char *mapset;
+    struct Colors colrules;
+    unsigned char *r, *g, *b, *set;
+    int *cur, i, j;
+
+    mapset = G_find_raster2(filename, "");
+    if (!mapset) {
+	G_warning(_("Raster map <%s> not found"), filename);
+	return;
+    }
+
+    r = (unsigned char *)G_malloc(cols);
+    g = (unsigned char *)G_malloc(cols);
+    b = (unsigned char *)G_malloc(cols);
+    set = (unsigned char *)G_malloc(cols);
+
+    Rast_read_colors(filename, mapset, &colrules);
+
+    cur = buff;
+
+    G_message(_("Translating colors from raster map <%s>..."),
+	      G_fully_qualified_name(filename, mapset));
+
+    for (i = 0; i < rows; i++) {
+	Rast_lookup_c_colors(cur, r, g, b, set, cols, &colrules);
+	G_percent(i, rows, 2);
+
+	for (j = 0; j < cols; j++) {
+	    if (set[j]) {
+		cur[j] =
+		    (r[j] & 0xff) | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) <<
+							    16);
+	    }
+	    else {
+		cur[j] = NO_DATA_COL;
+	    }
+	}
+
+	cur = &(cur[cols]);
+    }
+    G_percent(1, 1, 1);
+    
+    Rast_free_colors(&colrules);
+
+    G_free(r);
+    G_free(g);
+    G_free(b);
+
+    G_free(set);
+
+    return;
+}
+
+/*!
+   \brief Pack color table (floating-point map)
+
+   Passed a array of floats that will be converted from cell values
+   to packed colors (0xbbggrr) and float to int 
+   Floating point data not freed here, use: 
+   gsds_free_data_buff(id, ATTY_FLOAT)
+
+   \param filename raster map name
+   \param fbuf
+   \param ibuf
+   \param rows number of rows
+   \param cols number of cols
+ */
+void Gs_pack_colors_float(const char *filename, float *fbuf, int *ibuf,
+			  int rows, int cols)
+{
+    const char *mapset;
+    struct Colors colrules;
+    unsigned char *r, *g, *b, *set;
+    int i, j, *icur;
+    FCELL *fcur;
+
+    mapset = G_find_raster2(filename, "");
+    if (!mapset) {
+	G_warning(_("Raster map <%s> not found"), filename);
+	return;
+    }
+
+    r = (unsigned char *)G_malloc(cols);
+    g = (unsigned char *)G_malloc(cols);
+    b = (unsigned char *)G_malloc(cols);
+    set = (unsigned char *)G_malloc(cols);
+
+    Rast_read_colors(filename, mapset, &colrules);
+
+    fcur = fbuf;
+    icur = ibuf;
+
+    G_message(_("Translating colors from raster map <%s>..."),
+	      G_fully_qualified_name(filename, mapset));
+    
+    for (i = 0; i < rows; i++) {
+	Rast_lookup_f_colors(fcur, r, g, b, set, cols, &colrules);
+	G_percent(i, rows, 2);
+
+	for (j = 0; j < cols; j++) {
+	    if (set[j]) {
+		icur[j] =
+		    (r[j] & 0xff) | ((g[j] & 0xff) << 8) | ((b[j] & 0xff) <<
+							    16);
+	    }
+	    else {
+		icur[j] = NO_DATA_COL;
+	    }
+	}
+
+	icur = &(icur[cols]);
+	fcur = &(fcur[cols]);
+    }
+    G_percent(1, 1, 1);
+    
+    Rast_free_colors(&colrules);
+
+    G_free(r);
+    G_free(g);
+    G_free(b);
+    G_free(set);
+
+    return;
+}
+
+/*!
+   \brief Get categories/labels
+
+   Formats label as in d.what.rast -> (catval) catlabel 
+
+   \param filename raster map name
+   \param drow
+   \param dcol
+   \param catstr category string
+
+   \return 1 on success
+   \return 0 on failure
+ */
+int Gs_get_cat_label(const char *filename, int drow, int dcol, char *catstr)
+{
+    struct Categories cats;
+    const char *mapset;
+    CELL *buf;
+    DCELL *dbuf;
+    RASTER_MAP_TYPE map_type;
+    int fd = -1;
+
+    if ((mapset = G_find_raster2(filename, "")) == NULL) {
+	G_warning(_("Raster map <%s> not found"), filename);
+	return 0;
+    }
+
+    if (-1 != Rast_read_cats(filename, mapset, &cats)) {
+	fd = Rast_open_old(filename, mapset);
+	map_type = Rast_get_map_type(fd);
+
+	if (map_type == CELL_TYPE) {
+	    buf = Rast_allocate_c_buf();
+
+	    Rast_get_c_row(fd, buf, drow);
+	    if (Rast_is_c_null_value(&buf[dcol])) {
+		sprintf(catstr, "(NULL) %s",
+			Rast_get_c_cat(&buf[dcol], &cats));
+	    }
+	    else {
+		sprintf(catstr, "(%d) %s", buf[dcol],
+			Rast_get_c_cat(&buf[dcol], &cats));
+	    }
+
+	    G_free(buf);
+	}
+
+	else {
+	    /* fp map */
+	    dbuf = Rast_allocate_d_buf();
+
+	    Rast_get_d_row(fd, dbuf, drow);
+	    if (Rast_is_d_null_value(&dbuf[dcol])) {
+		sprintf(catstr, "(NULL) %s",
+			Rast_get_d_cat(&dbuf[dcol], &cats));
+	    }
+	    else {
+		sprintf(catstr, "(%g) %s", dbuf[dcol],
+			Rast_get_d_cat(&dbuf[dcol], &cats));
+	    }
+
+	    G_free(dbuf);
+	}
+    }
+    else {
+	strcpy(catstr, "no category label");
+	return 0;
+    }
+
+    /* TODO: may want to keep these around for multiple queries */
+    Rast_free_cats(&cats);
+
+    if (fd >= 0)
+	Rast_close(fd);
+
+    return (1);
+}
+
+/*!
+   \brief Save 3dview
+
+   \param vname view name
+   \param gv pointer to geoview struct
+   \param gd pointer to geodisplay struct
+   \param w current window
+   \param defsurf default geosurf struct
+
+   \return -1 on error
+   \return ?
+ */
+int Gs_save_3dview(const char *vname, geoview * gv, geodisplay * gd,
+		   struct Cell_head *w, geosurf * defsurf)
+{
+    const char *mapset;
+    struct G_3dview v;
+    float zmax, zmin;
+
+    GS_get_zrange(&zmin, &zmax, 0);
+
+    G_get_3dview_defaults(&v, w);
+    mapset = G_mapset();
+
+    if (mapset != NULL) {
+	if (defsurf) {
+	    if (defsurf->draw_mode & DM_WIRE_POLY) {
+		v.display_type = 3;
+	    }
+	    else if (defsurf->draw_mode & DM_WIRE ||
+		     defsurf->draw_mode & DM_COL_WIRE) {
+		v.display_type = 1;
+	    }
+	    else if (defsurf->draw_mode & DM_POLY) {
+		v.display_type = 2;
+	    }
+
+	    v.mesh_freq = defsurf->x_modw;	/* mesh resolution */
+	    v.poly_freq = defsurf->x_mod;	/* poly resolution */
+	    v.dozero = !(defsurf->nz_topo);
+	    v.colorgrid = (defsurf->draw_mode & DM_COL_WIRE) ? 1 : 0;
+	    v.shading = (defsurf->draw_mode & DM_GOURAUD) ? 1 : 0;
+	}
+
+	if (gv->infocus) {
+	    GS_v3eq(v.from_to[TO], gv->real_to);
+	    v.from_to[TO][Z] -= zmin;
+	    GS_v3mult(v.from_to[TO], gv->scale);
+	    v.from_to[TO][Z] *= gv->vert_exag;
+	}
+	else {
+	    GS_v3eq(v.from_to[TO], gv->from_to[TO]);
+	}
+
+	gsd_model2real(v.from_to[TO]);
+
+	GS_v3eq(v.from_to[FROM], gv->from_to[FROM]);
+	gsd_model2real(v.from_to[FROM]);
+
+	v.exag = gv->vert_exag;
+	v.fov = gv->fov / 10.;
+	v.twist = gv->twist;
+	v.fringe = 0;		/* not implemented here */
+
+	v.lightson = 1;		/* always true, curently */
+
+	if (gv->lights[0].position[W] == 1) {
+	    /* local */
+	    v.lightpos[X] = gv->lights[0].position[X];
+	    v.lightpos[Y] = gv->lights[0].position[Y];
+	    v.lightpos[Z] = gv->lights[0].position[Z];
+	    gsd_model2real(v.lightpos);
+	    v.lightpos[W] = 1.0;	/* local */
+	}
+	else {
+	    v.lightpos[X] = gv->lights[0].position[X];
+	    v.lightpos[Y] = gv->lights[0].position[Y];
+	    v.lightpos[Z] = gv->lights[0].position[Z];
+	    v.lightpos[W] = 0.0;	/* inf */
+	}
+
+	v.lightcol[0] = gv->lights[0].color[0];
+	v.lightcol[1] = gv->lights[0].color[1];
+	v.lightcol[2] = gv->lights[0].color[2];
+
+	v.ambient = (gv->lights[0].ambient[0] + gv->lights[0].ambient[1] +
+		     gv->lights[0].ambient[2]) / 3.;
+	v.shine = gv->lights[0].shine;
+
+	v.surfonly = 0;		/* N/A - now uses constant color */
+	strcpy((v.pgm_id), "Nvision-ALPHA!");
+
+	return (G_put_3dview(vname, mapset, &v, w));
+    }
+    else {
+	return (-1);
+    }
+}
+
+/*!
+   \brief Load 3dview
+
+   \param vname view name
+   \param gv pointer to geoview struct
+   \param gd pointer to geodisplay struct
+   \param w current window
+   \param defsurf default geosurf struct
+
+   \return 1
+ */
+int Gs_load_3dview(const char *vname, geoview * gv, geodisplay * gd,
+		   struct Cell_head *w, geosurf * defsurf)
+{
+    const char *mapset;
+    struct G_3dview v;
+    int ret = -1;
+    float pt[3];
+
+    mapset = G_find_file2("3d.view", vname, "");
+
+    if (mapset != NULL) {
+	ret = G_get_3dview(vname, mapset, &v);
+    }
+
+    if (ret >= 0) {
+	if (strcmp((v.pgm_id), "Nvision-ALPHA!")) {
+	    G_warning(_("View not saved by this program,"
+			"there may be some inconsistancies"));
+	}
+
+	/* set poly and mesh resolutions */
+	v.mesh_freq = (int)(v.mesh_freq * v.vwin.ns_res / w->ns_res);
+	v.poly_freq = (int)(v.poly_freq * v.vwin.ns_res / w->ns_res);
+
+	/* Set To and FROM positions */
+	/* TO */
+	pt[0] = (v.from_to[TO][X] - w->west) - w->ew_res / 2.;
+	pt[1] = (v.from_to[TO][Y] - w->south) - w->ns_res / 2.;
+	pt[2] = v.from_to[TO][Z];
+	GS_set_focus(pt);
+
+	/* FROM */
+	pt[0] = (float)v.from_to[FROM][X];
+	pt[1] = (float)v.from_to[FROM][Y];
+	pt[2] = (float)v.from_to[FROM][Z];
+	GS_moveto_real(pt);
+
+	if (defsurf) {
+	    int dmode = 0;
+
+	    GS_setall_drawres(v.poly_freq, v.poly_freq,
+			      v.mesh_freq, v.mesh_freq);
+
+	    while (v.display_type >= 10) {
+		/* globe stuff not used */
+		v.display_type -= 10;
+	    }
+
+	    /* set drawing modes */
+	    if (v.colorgrid) {
+		dmode |= DM_COL_WIRE;
+	    }
+
+	    if (v.shading) {
+		dmode |= DM_GOURAUD;
+	    }
+
+	    switch (v.display_type) {
+	    case 1:
+		dmode |= DM_WIRE;
+
+		break;
+	    case 2:
+		dmode |= DM_POLY;
+
+		break;
+	    case 3:
+		dmode |= DM_WIRE_POLY;
+
+		break;
+	    }
+	    GS_setall_drawmode(dmode);
+
+	    /* should also set nozeros here */
+	}
+
+	/* set exaggeration */
+	if (v.exag)
+	    GS_set_global_exag(v.exag);
+
+	/* Set FOV */
+	if (v.fov) {
+	    GS_set_fov((int)
+		       (v.fov > 0 ? v.fov * 10. + 0.5 : v.fov * 10. - 0.5));
+	}
+	else {
+	    /* TODO: do ortho */
+	}
+
+	/* Set twist */
+	if (v.twist)
+	    GS_set_twist((int)(v.twist > 0 ? v.twist + 0.5 : v.twist - 0.5));
+
+
+	/* TODO:  OK to here - need to unravel/reverse lights stuff*** */
+
+	if (v.lightson) {
+	    /* Lights are on */
+
+	    /* Light Position */
+	    gv->lights[0].position[X] = v.lightpos[X];
+	    gv->lights[0].position[Y] = v.lightpos[Y];
+	    gv->lights[0].position[Z] = v.lightpos[Z];
+
+	    /* Light Color */
+	    gv->lights[0].color[0] = v.lightcol[0];
+	    gv->lights[0].color[1] = v.lightcol[1];
+	    gv->lights[0].color[2] = v.lightcol[2];
+
+	    /* Light Shininess */
+	    gv->lights[0].shine = v.shine;
+
+	    /* Light Ambient */
+	    gv->lights[0].ambient[0] = gv->lights[0].ambient[1] =
+		gv->lights[0].ambient[2] = v.ambient * 3.;
+
+
+	}			/* Done with lights */
+
+
+	GS_alldraw_wire();
+
+    }				/* Done with file */
+    return (1);
+
+}
+
+/*!
+   \brief Update no_zero ranges for attribute (actually no_null now)
+
+   \param gs pointer to geosurf struct
+   \param desc attribute id (descriptor)
+
+   \return -1 on error
+   \return 1 on success
+ */
+int Gs_update_attrange(geosurf * gs, int desc)
+{
+    long size;
+    float min, max;
+    typbuff *tb;
+    struct BM *nm;
+    int found;
+
+    gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].range_nz =
+	0.0;
+
+    if (CONST_ATT == gs_get_att_src(gs, desc)) {
+	gs->att[desc].max_nz = gs->att[desc].min_nz = gs->att[desc].constant;
+	min = max = gs->att[desc].constant;
+	gs->att[desc].range_nz = 0.0;
+    }
+    else if (CF_COLOR_PACKED & gsds_get_changed(gs->att[desc].hdata)) {
+	gs->att[desc].max_nz = 0xFFFFFF;
+	gs->att[desc].min_nz = 0x010101;
+	gs->att[desc].range_nz = 0xFFFFFF;
+    }
+    else {
+	if (NULL == (tb = gsds_get_typbuff(gs->att[desc].hdata, 0))) {
+	    return (-1);
+	}
+
+	nm = tb->nm;
+
+	if (tb->ib) {
+	    int *p;
+
+	    size = gs->rows * gs->cols;
+	    p = tb->ib;
+	    INIT_MINMAX(p, nm, size, min, max, found);
+
+	    if (!found) {
+		/* all nulls! */
+		return (-1);
+	    }
+
+	    size = gs->rows * gs->cols;
+	    p = tb->ib;
+	    SET_MINMAX(p, nm, size, min, max);
+	}
+	else if (tb->sb) {
+	    short *p;
+
+	    size = gs->rows * gs->cols;
+	    p = tb->sb;
+	    INIT_MINMAX(p, nm, size, min, max, found);
+
+	    if (!found) {
+		/* all nulls! */
+		return (-1);
+	    }
+
+	    size = gs->rows * gs->cols;
+	    p = tb->sb;
+	    SET_MINMAX(p, nm, size, min, max);
+	}
+	else if (tb->cb) {
+	    char *p;
+
+	    size = gs->rows * gs->cols;
+	    p = (char *)tb->cb;
+	    INIT_MINMAX(p, nm, size, min, max, found);
+
+	    if (!found) {
+		/* all nulls! */
+		return (-1);
+	    }
+
+	    size = gs->rows * gs->cols;
+	    p = (char *)tb->cb;
+	    SET_MINMAX(p, nm, size, min, max);
+	}
+	else if (tb->fb) {
+	    float *p;
+
+	    size = gs->rows * gs->cols;
+	    p = tb->fb;
+	    INIT_MINMAX(p, nm, size, min, max, found);
+
+	    if (!found) {
+		/* all nulls! */
+		return (-1);
+	    }
+
+	    size = gs->rows * gs->cols;
+	    p = tb->fb;
+	    SET_MINMAX(p, nm, size, min, max);
+	}
+
+	gs->att[desc].max_nz = max;
+	gs->att[desc].min_nz = min;
+	gs->att[desc].range_nz = gs->att[desc].max_nz - gs->att[desc].min_nz;
+    }
+
+    if (ATT_TOPO == desc) {
+	gs->zmin = min;
+	gs->zmax = max;
+	gs->zrange = gs->zmax - gs->zmin;
+	gs->zminmasked = gs->zmin;
+	gs->zmax_nz = gs->zmax;
+	gs->zmin_nz = gs->zmin;
+	gs->zrange_nz = gs->zmax_nz - gs->zmin_nz;
+    }
+
+    G_debug(3, "Gs_update_attrange(): min=%f max=%f", gs->zmin, gs->zmax);
+
+    return (1);
+}

Copied: grass/trunk/lib/ogsf/gs_util.c (from rev 62429, grass/trunk/lib/ogsf/GS_util.c)
===================================================================
--- grass/trunk/lib/ogsf/gs_util.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gs_util.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,486 @@
+/*!
+   \file GS_util.c
+
+   \brief OGSF library - loading and manipulating surfaces
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown USACERL, GMSL/University of Illinois
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <stdlib.h>
+#include <math.h>
+#include <string.h>
+
+#include <grass/gis.h>
+#include <grass/ogsf.h>
+
+/*!
+   \brief Calculate distance between 2 coordinates
+
+   Units is one of:
+   - "meters",
+   - "miles",
+   - "kilometers",
+   - "feet",
+   - "yards",
+   - "nmiles" (nautical miles),
+   - "rods",
+   - "inches",
+   - "centimeters",
+   - "millimeters",
+   - "micron",
+   - "nanometers",
+   - "cubits",
+   - "hands",
+   - "furlongs",
+   - "chains"
+
+   Default is meters.
+
+   \param from starting point
+   \param to ending point
+   \param units map units
+
+   \return distance between two geographic coordinates in current projection
+ */
+double GS_geodistance(double *from, double *to, const char *units)
+{
+    double meters;
+
+    meters = Gs_distance(from, to);
+
+    if (!units) {
+	return (meters);
+    }
+
+    if (strcmp(units, "meters") == 0) {
+	return (meters);
+    }
+
+    if (strcmp(units, "miles") == 0) {
+	return (meters * .0006213712);
+    }
+
+    if (strcmp(units, "kilometers") == 0) {
+	return (meters * .001);
+    }
+
+    if (strcmp(units, "feet") == 0) {
+	return (meters * 3.280840);
+    }
+
+    if (strcmp(units, "yards") == 0) {
+	return (meters * 1.093613);
+    }
+
+    if (strcmp(units, "rods") == 0) {
+	return (meters * .1988388);
+    }
+
+    if (strcmp(units, "inches") == 0) {
+	return (meters * 39.37008);
+    }
+
+    if (strcmp(units, "centimeters") == 0) {
+	return (meters * 100.0);
+    }
+
+    if (strcmp(units, "millimeters") == 0) {
+	return (meters * 1000.0);
+    }
+
+    if (strcmp(units, "micron") == 0) {
+	return (meters * 1000000.0);
+    }
+
+    if (strcmp(units, "nanometers") == 0) {
+	return (meters * 1000000000.0);
+    }
+
+    if (strcmp(units, "cubits") == 0) {
+	return (meters * 2.187227);
+    }
+
+    if (strcmp(units, "hands") == 0) {
+	return (meters * 9.842520);
+    }
+
+    if (strcmp(units, "furlongs") == 0) {
+	return (meters * .004970970);
+    }
+
+    if (strcmp(units, "nmiles") == 0) {
+	/* nautical miles */
+	return (meters * .0005399568);
+    }
+
+    if (strcmp(units, "chains") == 0) {
+	return (meters * .0497097);
+    }
+
+    return (meters);
+}
+
+/*!
+   \brief Calculate distance
+
+   \param from 'from' point (X,Y,Z)
+   \param to 'to' point (X,Y,Z)
+
+   \return distance
+ */
+float GS_distance(float *from, float *to)
+{
+    float x, y, z;
+
+    x = from[X] - to[X];
+    y = from[Y] - to[Y];
+    z = from[Z] - to[Z];
+
+    return (float)sqrt(x * x + y * y + z * z);
+}
+
+/*!
+   \brief Calculate distance in plane
+
+   \param from 'from' point (X,Y)
+   \param to 'to' point (X,Y)
+
+   \return distance
+ */
+float GS_P2distance(float *from, float *to)
+{
+    float x, y;
+
+    x = from[X] - to[X];
+    y = from[Y] - to[Y];
+
+    return (float)sqrt(x * x + y * y);
+}
+
+/*!
+   \brief Copy vector values
+
+   v1 = v2
+
+   \param[out] v1 first vector
+   \param v2 second vector
+ */
+void GS_v3eq(float *v1, float *v2)
+{
+    v1[X] = v2[X];
+    v1[Y] = v2[Y];
+    v1[Z] = v2[Z];
+
+    return;
+}
+
+/*!
+   \brief Sum vectors
+
+   v1 += v2
+
+   \param[in,out] v1 first vector
+   \param v2 second vector
+ */
+void GS_v3add(float *v1, float *v2)
+{
+    v1[X] += v2[X];
+    v1[Y] += v2[Y];
+    v1[Z] += v2[Z];
+
+    return;
+}
+
+/*!
+   \brief Subtract vectors
+
+   v1 -= v2
+
+   \param[in,out] v1 first vector
+   \param v2 second vector
+ */
+void GS_v3sub(float *v1, float *v2)
+{
+    v1[X] -= v2[X];
+    v1[Y] -= v2[Y];
+    v1[Z] -= v2[Z];
+
+    return;
+}
+
+/*!
+   \brief Multiple vectors
+
+   v1 *= k
+
+   \param[in,out] v1 vector
+   \param k multiplicator
+ */
+void GS_v3mult(float *v1, float k)
+{
+    v1[X] *= k;
+    v1[Y] *= k;
+    v1[Z] *= k;
+
+    return;
+}
+
+/*!
+   \brief Change v1 so that it is a unit vector (2D)
+
+   \param[in,out] v1 vector
+
+   \return 0 if magnitude of v1 is zero
+   \return 1 if magnitude of v1 > 0
+ */
+int GS_v3norm(float *v1)
+{
+    float n;
+
+    n = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y] + v1[Z] * v1[Z]);
+
+    if (n == 0.0) {
+	return (0);
+    }
+
+    v1[X] /= n;
+    v1[Y] /= n;
+    v1[Z] /= n;
+
+    return (1);
+}
+
+/*!
+   \brief Change v1 so that it is a unit vector (3D)
+
+   \param[in,out] v1 vector
+
+   \return 0 if magnitude of v1 is zero
+   \return 1 if magnitude of v1 > 0
+ */
+int GS_v2norm(float *v1)
+{
+    float n;
+
+    n = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y]);
+
+    if (n == 0.0) {
+	return (0);
+    }
+
+    v1[X] /= n;
+    v1[Y] /= n;
+
+    return (1);
+}
+
+/*!
+   \brief Changes v1 so that it is a unit vector
+
+   \param dv1 vector
+
+   \return 0 if magnitude of dv1 is zero
+   \return 1 if magnitude of dv1 > 0
+ */
+int GS_dv3norm(double *dv1)
+{
+    double n;
+
+    n = sqrt(dv1[X] * dv1[X] + dv1[Y] * dv1[Y] + dv1[Z] * dv1[Z]);
+
+    if (n == 0.0) {
+	return (0);
+    }
+
+    dv1[X] /= n;
+    dv1[Y] /= n;
+    dv1[Z] /= n;
+
+    return (1);
+}
+
+
+/*!
+   \brief Change v2 so that v1v2 is a unit vector
+
+   \param v1 first vector
+   \param v2[in,out] second vector
+
+   \return 0 if magnitude of dx is zero
+   \return 1 if magnitude of dx > 0
+ */
+int GS_v3normalize(float *v1, float *v2)
+{
+    float n, dx, dy, dz;
+
+    dx = v2[X] - v1[X];
+    dy = v2[Y] - v1[Y];
+    dz = v2[Z] - v1[Z];
+    n = sqrt(dx * dx + dy * dy + dz * dz);
+
+    if (n == 0.0) {
+	return (0);
+    }
+
+    v2[X] = v1[X] + dx / n;
+    v2[Y] = v1[Y] + dy / n;
+    v2[Z] = v1[Z] + dz / n;
+
+    return (1);
+}
+
+
+/*!
+   \brief Get a normalized direction from v1 to v2, store in v3
+
+   \param v1 first vector
+   \param v2 second vector
+   \param[out] v3 output vector
+
+   \return 0 if magnitude of dx is zero
+   \return 1 if magnitude of dx > 0
+ */
+int GS_v3dir(float *v1, float *v2, float *v3)
+{
+    float n, dx, dy, dz;
+
+    dx = v2[X] - v1[X];
+    dy = v2[Y] - v1[Y];
+    dz = v2[Z] - v1[Z];
+    n = sqrt(dx * dx + dy * dy + dz * dz);
+
+    if (n == 0.0) {
+	v3[X] = v3[Y] = v3[Z] = 0.0;
+	return (0);
+    }
+
+    v3[X] = dx / n;
+    v3[Y] = dy / n;
+    v3[Z] = dz / n;
+
+    return (1);
+}
+
+
+/*!
+   \brief Get a normalized direction from v1 to v2, store in v3 (2D)
+
+   \param v1 first vector
+   \param v2 second vector
+   \param[out] v3 output vector
+
+   \return 0 if magnitude of dx is zero
+   \return 1 if magnitude of dx > 0
+ */
+void GS_v2dir(float *v1, float *v2, float *v3)
+{
+    float n, dx, dy;
+
+    dx = v2[X] - v1[X];
+    dy = v2[Y] - v1[Y];
+    n = sqrt(dx * dx + dy * dy);
+
+    v3[X] = dx / n;
+    v3[Y] = dy / n;
+
+    return;
+}
+
+/*!
+   \brief Get the cross product v3 = v1 cross v2
+
+   \param v1 first vector
+   \param v2 second vector
+   \param[out] v3 output vector
+ */
+void GS_v3cross(float *v1, float *v2, float *v3)
+{
+    v3[X] = (v1[Y] * v2[Z]) - (v1[Z] * v2[Y]);
+    v3[Y] = (v1[Z] * v2[X]) - (v1[X] * v2[Z]);
+    v3[Z] = (v1[X] * v2[Y]) - (v1[Y] * v2[X]);
+
+    return;
+}
+
+/*!
+   \brief Magnitude of vector
+
+   \param v1 vector
+   \param[out] mag magnitude value
+ */
+void GS_v3mag(float *v1, float *mag)
+{
+    *mag = sqrt(v1[X] * v1[X] + v1[Y] * v1[Y] + v1[Z] * v1[Z]);
+
+    return;
+}
+
+/*!
+   \brief ADD
+
+   Initialize by calling with a number nhist to represent number of
+   previous entrys to check, then call with zero as nhist
+
+   \param p1 first point
+   \param p2 second point
+   \param nhist ?
+
+   \return -1 on error
+   \return -2
+   \return 1
+   \return 9
+ */
+int GS_coordpair_repeats(float *p1, float *p2, int nhist)
+{
+    static float *entrys = NULL;
+    static int next = 0;
+    static int len = 0;
+    int i;
+
+    if (nhist) {
+	if (entrys) {
+	    G_free(entrys);
+	}
+
+	entrys = (float *)G_malloc(4 * nhist * sizeof(float));
+
+	if (!entrys)
+	    return (-1);
+
+	len = nhist;
+	next = 0;
+    }
+
+    if (!len) {
+	return (-2);
+    }
+
+    for (i = 0; i < next; i += 4) {
+	if (entrys[i] == p1[0] && entrys[i + 1] == p1[1]
+	    && entrys[i + 2] == p2[0] && entrys[i + 3] == p2[1]) {
+	    return (1);
+	}
+    }
+
+    if (len == next / 4) {
+	next = 0;
+    }
+
+    entrys[next] = p1[0];
+    entrys[next + 1] = p1[1];
+    entrys[next + 2] = p2[0];
+    entrys[next + 3] = p2[1];
+    next += 4;
+
+    return (0);
+}

Copied: grass/trunk/lib/ogsf/gsx.c (from rev 62429, grass/trunk/lib/ogsf/GSX.c)
===================================================================
--- grass/trunk/lib/ogsf/gsx.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gsx.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,70 @@
+/*!
+   \file GSX.c
+
+   \brief OGSF library - loading and manipulating surfaces
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown USACERL (December 1993)
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <grass/ogsf.h>
+
+void (*Cxl_func) ();
+void (*Swap_func) ();
+
+static int Cxl = 0;
+
+/*!
+   \brief Check for cancel
+
+   \return code
+ */
+int GS_check_cancel(void)
+{
+    Cxl_func();
+
+    return (Cxl);
+}
+
+/*!
+   \brief Set cancel
+ */
+void GS_set_cancel(int c)
+{
+    Cxl = c;
+
+    return;
+}
+
+/*!
+   \brief Set cxl function
+
+   \param pointer to function
+ */
+void GS_set_cxl_func(void (*f) (void))
+{
+    Cxl_func = f;
+
+    return;
+}
+
+/*!
+   \brief Set swap function
+
+   \param pointer to function
+ */
+void GS_set_swap_func(void (*f) (void))
+{
+    Swap_func = f;
+
+    return;
+}

Copied: grass/trunk/lib/ogsf/gv2.c (from rev 62429, grass/trunk/lib/ogsf/GV2.c)
===================================================================
--- grass/trunk/lib/ogsf/gv2.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gv2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,601 @@
+/*!
+   \file lib/ogsf/GV2.c
+
+   \brief OGSF library - loading and manipulating vector sets (higher level functions)
+
+   (C) 1999-2008, 2011 by the GRASS Development Team
+
+   This program is free software under the GNU General Public License
+   (>=v2).  Read the file COPYING that comes with GRASS for details.
+
+   \author Bill Brown USACERL, GMSL/University of Illinois
+   \author Updated by Martin landa <landa.martin gmail.com>
+   (doxygenized in May 2008, thematic mapping in June 2011)
+ */
+
+#include <stdlib.h>
+#include <string.h>
+
+#include <grass/gis.h>
+#include <grass/ogsf.h>
+
+#include "gsget.h"
+
+static int Vect_ID[MAX_VECTS];
+static int Next_vect = 0;
+
+/*!
+   \brief Check if vector set exists
+
+   \param id vector set id
+
+   \return 0 not found
+   \return 1 found
+ */
+int GV_vect_exists(int id)
+{
+    int i, found = 0;
+
+    G_debug(3, "GV_vect_exists");
+
+    if (NULL == gv_get_vect(id)) {
+	return (0);
+    }
+
+    for (i = 0; i < Next_vect && !found; i++) {
+	if (Vect_ID[i] == id) {
+	    found = 1;
+	}
+    }
+
+    return (found);
+}
+
+/*!
+   \brief Register new vector set
+
+   \return vector set id
+   \return -1 on error
+ */
+int GV_new_vector(void)
+{
+    geovect *nv;
+
+    if (Next_vect < MAX_VECTS) {
+	nv = gv_get_new_vect();
+	gv_set_defaults(nv);
+	Vect_ID[Next_vect] = nv->gvect_id;
+	++Next_vect;
+
+	G_debug(3, "GV_new_vector(): id=%d", nv->gvect_id);
+
+	return nv->gvect_id;
+    }
+
+    return -1;
+}
+
+/*!
+   \brief Get number of available vector sets
+
+   \return number of vector sets
+ */
+int GV_num_vects(void)
+{
+    return (gv_num_vects());
+}
+
+/*!
+   \brief Get list of vector sets
+
+   Must free when no longer needed!
+
+   \param numvects number of vector sets
+
+   \return pointer to list of point sets
+   \return NULL on error
+ */
+int *GV_get_vect_list(int *numvects)
+{
+    int i, *ret;
+
+    *numvects = Next_vect;
+
+    if (Next_vect) {
+	ret = (int *)G_malloc(Next_vect * sizeof(int));
+	if (!ret) {
+	    return (NULL);
+	}
+
+	for (i = 0; i < Next_vect; i++) {
+	    ret[i] = Vect_ID[i];
+	}
+
+	return (ret);
+    }
+
+    return (NULL);
+}
+
+/*!
+   \brief Delete vector set from list
+
+   \param id vector set id
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GV_delete_vector(int id)
+{
+    int i, j, found = 0;
+
+    G_debug(3, "GV_delete_vect");
+
+    if (GV_vect_exists(id)) {
+	gv_delete_vect(id);
+
+	for (i = 0; i < Next_vect && !found; i++) {
+	    if (Vect_ID[i] == id) {
+		found = 1;
+
+		for (j = i; j < Next_vect; j++) {
+		    Vect_ID[j] = Vect_ID[j + 1];
+		}
+	    }
+	}
+
+	if (found) {
+	    --Next_vect;
+	    return (1);
+	}
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Load vector set
+
+   Check to see if handle already loaded, if so - free before loading
+   new for now, always load to memory
+
+   \todo Load file handle & ready for reading instead of using
+   memory
+
+   \param id vector set id
+   \param filename filename
+
+   \return -1 on error (invalid vector set id)
+   \return 1 on success
+ */
+int GV_load_vector(int id, const char *filename)
+{
+    geovect *gv;
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return (-1);
+    }
+
+    if (gv->lines) {
+	gv_free_vectmem(gv);
+    }
+
+    gv->filename = G_store(filename);
+
+    if ((gv->lines = Gv_load_vect(filename, &(gv->n_lines)))) {
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get vector map name
+
+   Note: char array is allocated by G_store()
+
+   \param id vector set id
+   \param filename &filename
+
+   \return -1 on error (invalid vector set id)
+   \return 1 on success
+ */
+int GV_get_vectname(int id, char **filename)
+{
+    geovect *gv;
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return (-1);
+    }
+
+    *filename = G_store(gv->filename);
+
+    return (1);
+}
+
+/*!
+   \brief Set vector style
+
+   \param id vector set id
+   \param mem non-zero for use memory
+   \param color color value
+   \param width line width
+   \param flat non-zero for flat mode
+
+   \return -1 on error (invalid vector set id)
+   \return 1 on success
+ */
+int GV_set_style(int id, int mem, int color, int width, int flat)
+{
+    geovect *gv;
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return -1;
+    }
+
+    gv->use_mem = mem;
+    gv->flat_val = flat;
+    gv->style->color = color;
+    gv->style->width = width;
+
+    return 1;
+}
+
+
+/*!
+   \brief Get vector style
+
+   \param id vector set id
+   \param[out] mem non-zero for use memory
+   \param[out] color color value
+   \param[out] width line width
+   \param[out] flat non-zero for flat mode
+
+   \return -1 on error (invalid vector set id)
+   \return 1 on success
+ */
+int GV_get_style(int id, int *mem, int *color, int *width, int *flat)
+{
+    geovect *gv;
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return -1;
+    }
+
+    *mem = gv->use_mem;
+    *color = gv->style->color;
+    *width = gv->style->width;
+    *flat = gv->flat_val;
+
+    return 1;
+}
+
+/*!
+   \brief Set vector set style for thematic mapping
+   
+   Updates also style for each geoline.
+   
+   \param id vector set id
+   \param layer layer number for thematic mapping
+   \param color color column name
+   \param width width column name
+   \param colors pointer to Colors structure or NULL
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GV_set_style_thematic(int id, int layer, const char* color, const char* width,
+			  struct Colors *color_rules)
+{
+    geovect *gv;
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return -1;
+    }
+
+    if(!gv->tstyle)
+	gv->tstyle = (gvstyle_thematic *)G_malloc(sizeof(gvstyle_thematic));
+    G_zero(gv->tstyle, sizeof(gvstyle_thematic));
+    
+    gv->tstyle->active = 1;
+    gv->tstyle->layer = layer;
+    if (color)
+	gv->tstyle->color_column = G_store(color);
+    if (width)
+	gv->tstyle->width_column = G_store(width);
+
+    Gv_load_vect_thematic(gv, color_rules);
+
+    return 1;
+}
+
+/*!
+   \brief Make style for thematic mapping inactive
+   
+   \param id vector set id
+
+   \return 1 on success
+   \return -1 on error (point set not found)
+ */
+int GV_unset_style_thematic(int id)
+{
+    geovect *gv;
+
+    G_debug(4, "GV_unset_style_thematic(): id=%d", id);
+
+    if (NULL == (gv = gv_get_vect(id))) {
+	return -1;
+    }
+
+    if (gv->tstyle) {
+	gv->tstyle->active = 0;
+    }
+
+    return 1;
+}
+
+/*!
+   \brief Set trans ?
+
+   \param id vector set id
+   \param xtrans,ytrans,ztrans x/y/z trans values
+ */
+void GV_set_trans(int id, float xtrans, float ytrans, float ztrans)
+{
+    geovect *gv;
+
+    G_debug(3, "GV_set_trans");
+
+    gv = gv_get_vect(id);
+
+    if (gv) {
+	gv->x_trans = xtrans;
+	gv->y_trans = ytrans;
+	gv->z_trans = ztrans;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get trans ?
+
+   \param id vector set id
+   \param[out] xtrans,ytrans,ztrans x/y/z trans values
+ */
+int GV_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
+{
+    geovect *gv;
+
+    gv = gv_get_vect(id);
+
+    if (gv) {
+	*xtrans = gv->x_trans;
+	*ytrans = gv->y_trans;
+	*ztrans = gv->z_trans;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Select surface identified by hs to have vector identified
+   by hv draped over it
+
+   \param hv vector set id
+   \param hs surface id
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GV_select_surf(int hv, int hs)
+{
+    geovect *gv;
+
+    if (GV_surf_is_selected(hv, hs)) {
+	return (1);
+    }
+
+    gv = gv_get_vect(hv);
+
+    if (gv && GS_surf_exists(hs)) {
+	gv->drape_surf_id[gv->n_surfs] = hs;
+	gv->n_surfs += 1;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Unselect surface
+
+   \param hv vector set id
+   \param hs surface id
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GV_unselect_surf(int hv, int hs)
+{
+    geovect *gv;
+    int i, j;
+
+    if (!GV_surf_is_selected(hv, hs)) {
+	return (1);
+    }
+
+    gv = gv_get_vect(hv);
+
+    if (gv) {
+	for (i = 0; i < gv->n_surfs; i++) {
+	    if (gv->drape_surf_id[i] == hs) {
+		for (j = i; j < gv->n_surfs - 1; j++) {
+		    gv->drape_surf_id[j] = gv->drape_surf_id[j + 1];
+		}
+
+		gv->n_surfs -= 1;
+
+		return (1);
+	    }
+	}
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Check if surface is selected
+
+   \param hv vector set id
+   \param hs surface id
+
+   \return 1 selected
+   \return 0 not selected
+ */
+int GV_surf_is_selected(int hv, int hs)
+{
+    int i;
+    geovect *gv;
+
+    gv = gv_get_vect(hv);
+
+    if (gv) {
+	for (i = 0; i < gv->n_surfs; i++) {
+	    if (hs == gv->drape_surf_id[i]) {
+		return (1);
+	    }
+	}
+    }
+
+    return (0);
+}
+
+/*!
+   \brief Draw vector set
+
+   \param vid vector set id
+ */
+void GV_draw_vect(int vid)
+{
+    geosurf *gs;
+    geovect *gv;
+    int i;
+
+    gv = gv_get_vect(vid);
+
+    if (gv) {
+	for (i = 0; i < gv->n_surfs; i++) {
+	    gs = gs_get_surf(gv->drape_surf_id[i]);
+
+	    if (gs) {
+		gvd_vect(gv, gs, 0);
+	    }
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all loaded vector sets
+ */
+void GV_alldraw_vect(void)
+{
+    int id;
+
+    for (id = 0; id < Next_vect; id++) {
+	GV_draw_vect(Vect_ID[id]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw vector set (fast mode)
+
+   \todo Seems to be broken, nothing is drawn
+
+   \param vid vector set id
+ */
+void GV_draw_fastvect(int vid)
+{
+    geosurf *gs;
+    geovect *gv;
+    int i;
+
+    gv = gv_get_vect(vid);
+
+    if (gv) {
+	for (i = 0; i < gv->n_surfs; i++) {
+	    gs = gs_get_surf(gv->drape_surf_id[i]);
+
+	    if (gs) {
+		gvd_vect(gv, gs, 1);
+	    }
+	}
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all loaded vector sets (fast mode)
+ */
+void GV_alldraw_fastvect(void)
+{
+    int id;
+
+    for (id = 0; id < Next_vect; id++) {
+	GV_draw_fastvect(Vect_ID[id]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Set client data
+
+   \param id vector set id
+   \param clientd pointer to client data
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GV_Set_ClientData(int id, void *clientd)
+{
+    geovect *gv;
+
+    gv = gv_get_vect(id);
+    if (gv) {
+	gv->clientdata = clientd;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get client data
+
+   \param id vector set id
+
+   \return pointer to client data
+   \return NULL on error
+ */
+void *GV_Get_ClientData(int id)
+{
+    geovect *gv;
+
+    gv = gv_get_vect(id);
+
+    if (gv) {
+	return (gv->clientdata);
+    }
+
+    return (NULL);
+}

Copied: grass/trunk/lib/ogsf/gv3.c (from rev 62429, grass/trunk/lib/ogsf/Gv3.c)
===================================================================
--- grass/trunk/lib/ogsf/gv3.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gv3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,419 @@
+/*!
+   \file lib/ogsf/Gv3.c
+
+   \brief OGSF library - loading vector sets (lower level functions)
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008, 2011 by the GRASS Development Team
+
+   This program is free software under the GNU General Public License
+   (>=v2). Read the file COPYING that comes with GRASS for details.
+
+   \author Bill Brown USACERL (December 1993)
+   \author Updated by Martin Landa <landa.martin gmail.com>
+   (doxygenized in May 2008, thematic mapping in August 2011)
+ */
+
+#include <stdlib.h>
+
+#include <grass/gis.h>
+#include <grass/colors.h>
+#include <grass/raster.h>
+#include <grass/vector.h>
+#include <grass/dbmi.h>
+#include <grass/glocale.h>
+#include <grass/ogsf.h>
+
+/*
+   #define TRAK_MEM
+*/
+
+#ifdef TRAK_MEM
+static int Tot_mem = 0;
+#endif
+
+/*!
+   \brief Load vector map to memory
+
+   The other alternative may be to load to a tmp file
+
+   \param grassname vector map name
+   \param[out] number of loaded features
+
+   \return pointer to geoline struct
+   \return NULL on failure
+ */
+geoline *Gv_load_vect(const char *grassname, int *nlines)
+{
+    struct Map_info map;
+    struct line_pnts *points;
+    struct line_cats *Cats = NULL;
+    geoline *top, *gln, *prev;
+    int np, i, n, nareas, nl = 0, area, type, is3d;
+    struct Cell_head wind;
+    float vect[2][3];
+    const char *mapset;
+
+    mapset = G_find_vector2(grassname, "");
+    if (!mapset) {
+	G_warning(_("Vector map <%s> not found"), grassname);
+	return NULL;
+    }
+
+    Vect_set_open_level(2);
+    if (Vect_open_old(&map, grassname, "") == -1) {
+	G_warning(_("Unable to open vector map <%s>"),
+		  G_fully_qualified_name(grassname, mapset));
+	return NULL;
+    }
+
+    top = gln = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
+    if (!top) {
+	return NULL;
+    }
+
+    prev = top;
+
+#ifdef TRAK_MEM
+    Tot_mem += sizeof(geoline);
+#endif
+
+    points = Vect_new_line_struct();
+    Cats = Vect_new_cats_struct();
+
+    G_get_set_window(&wind);
+    Vect_set_constraint_region(&map, wind.north, wind.south, wind.east,
+			       wind.west, PORT_DOUBLE_MAX, -PORT_DOUBLE_MAX);
+
+    is3d = Vect_is_3d(&map);
+
+    /* Read areas */
+    n = Vect_get_num_areas(&map);
+    nareas = 0;
+    G_debug(3, "Reading vector areas (nareas = %d)", n);
+    for (area = 1; area <= n; area++) {
+	G_debug(3, " area %d", area);
+	Vect_get_area_points(&map, area, points);
+	if (points->n_points < 3)
+	    continue;
+
+	/* initialize style */
+	gln->highlighted = 0;
+
+	gln->type = OGSF_POLYGON;
+	gln->npts = np = points->n_points;
+	G_debug(3, "  np = %d", np);
+
+	if (is3d) {
+	    gln->dims = 3;
+	    gln->p3 = (Point3 *) G_calloc(np, sizeof(Point3));	/* G_fatal_error */
+	    if (!gln->p3) {
+		return (NULL);
+	    }
+#ifdef TRAK_MEM
+	    Tot_mem += (np * sizeof(Point3));
+#endif
+	}
+	else {
+	    gln->dims = 2;
+	    gln->p2 = (Point2 *) G_calloc(np, sizeof(Point2));	/* G_fatal_error */
+	    if (!gln->p2) {
+		return (NULL);
+	    }
+#ifdef TRAK_MEM
+	    Tot_mem += (np * sizeof(Point2));
+#endif
+	}
+
+	for (i = 0; i < np; i++) {
+	    if (is3d) {
+		gln->p3[i][X] = points->x[i];
+		gln->p3[i][Y] = points->y[i];
+		gln->p3[i][Z] = points->z[i];
+	    }
+	    else {
+		gln->p2[i][X] = points->x[i];
+		gln->p2[i][Y] = points->y[i];
+	    }
+	}
+	/* Calc normal (should be average) */
+	if (is3d) {
+	    vect[0][X] = (float)(gln->p3[0][X] - gln->p3[1][X]);
+	    vect[0][Y] = (float)(gln->p3[0][Y] - gln->p3[1][Y]);
+	    vect[0][Z] = (float)(gln->p3[0][Z] - gln->p3[1][Z]);
+	    vect[1][X] = (float)(gln->p3[2][X] - gln->p3[1][X]);
+	    vect[1][Y] = (float)(gln->p3[2][Y] - gln->p3[1][Y]);
+	    vect[1][Z] = (float)(gln->p3[2][Z] - gln->p3[1][Z]);
+	    GS_v3cross(vect[1], vect[0], gln->norm);
+
+	}
+
+	gln->cats = NULL;
+	gln->next = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
+	if (!gln->next) {
+	    return (NULL);
+	}
+
+#ifdef TRAK_MEM
+	Tot_mem += sizeof(geoline);
+#endif
+
+	prev = gln;
+	gln = gln->next;
+	nareas++;
+    }
+    G_debug(3, "%d areas loaded", nareas);
+
+    /* Read all lines */
+    G_debug(3, "Reading vector lines ...");
+    while (-1 < (type = Vect_read_next_line(&map, points, Cats))) {
+	G_debug(3, "line type = %d", type);
+	if (type & (GV_LINES | GV_FACE)) {
+	    if (type & (GV_LINES)) {
+		gln->type = OGSF_LINE;
+	    }
+	    else {
+		gln->type = OGSF_POLYGON;
+		/* Vect_append_point ( points, points->x[0], points->y[0], points->z[0] ); */
+	    }
+
+	    /* initialize style */
+	    gln->highlighted = 0;
+
+	    gln->npts = np = points->n_points;
+	    G_debug(3, "  np = %d", np);
+
+	    if (is3d) {
+		gln->dims = 3;
+		gln->p3 = (Point3 *) G_calloc(np, sizeof(Point3));	/* G_fatal_error */
+		if (!gln->p3) {
+		    return (NULL);
+		}
+#ifdef TRAK_MEM
+		Tot_mem += (np * sizeof(Point3));
+#endif
+	    }
+	    else {
+		gln->dims = 2;
+		gln->p2 = (Point2 *) G_calloc(np, sizeof(Point2));	/* G_fatal_error */
+		if (!gln->p2) {
+		    return (NULL);
+		}
+#ifdef TRAK_MEM
+		Tot_mem += (np * sizeof(Point2));
+#endif
+	    }
+
+	    for (i = 0; i < np; i++) {
+		if (is3d) {
+		    gln->p3[i][X] = points->x[i];
+		    gln->p3[i][Y] = points->y[i];
+		    gln->p3[i][Z] = points->z[i];
+		}
+		else {
+		    gln->p2[i][X] = points->x[i];
+		    gln->p2[i][Y] = points->y[i];
+		}
+	    }
+	    /* Calc normal (should be average) */
+	    if (is3d && gln->type == OGSF_POLYGON) {
+		vect[0][X] = (float)(gln->p3[0][X] - gln->p3[1][X]);
+		vect[0][Y] = (float)(gln->p3[0][Y] - gln->p3[1][Y]);
+		vect[0][Z] = (float)(gln->p3[0][Z] - gln->p3[1][Z]);
+		vect[1][X] = (float)(gln->p3[2][X] - gln->p3[1][X]);
+		vect[1][Y] = (float)(gln->p3[2][Y] - gln->p3[1][Y]);
+		vect[1][Z] = (float)(gln->p3[2][Z] - gln->p3[1][Z]);
+		GS_v3cross(vect[1], vect[0], gln->norm);
+		G_debug(3, "norm %f %f %f", gln->norm[0], gln->norm[1],
+			gln->norm[2]);
+	    }
+
+	    /* Store category info for thematic display */
+	    if (Cats->n_cats > 0) {
+		gln->cats = Cats;
+		Cats = Vect_new_cats_struct();
+	    }
+	    else {
+		gln->cats = NULL;
+		Vect_reset_cats(Cats);
+	    }
+
+	    gln->next = (geoline *) G_malloc(sizeof(geoline));	/* G_fatal_error */
+	    if (!gln->next) {
+		return (NULL);
+	    }
+#ifdef TRAK_MEM
+	    Tot_mem += sizeof(geoline);
+#endif
+
+	    prev = gln;
+	    gln = gln->next;
+	    nl++;
+	}
+    }
+    G_debug(3, "%d lines loaded", nl);
+
+    nl += nareas;
+
+    prev->next = NULL;
+    G_free(gln);
+
+#ifdef TRAK_MEM
+    Tot_mem -= sizeof(geoline);
+#endif
+
+    Vect_close(&map);
+
+    if (!nl) {
+	G_warning(_("No features from vector map <%s> fall within current region"),
+		  G_fully_qualified_name(grassname, mapset));
+	return (NULL);
+    }
+    else {
+	G_message(_("Vector map <%s> loaded (%d features)"),
+		  G_fully_qualified_name(grassname, mapset), nl);
+    }
+
+    *nlines = nl;
+
+#ifdef TRAK_MEM
+    G_debug(3, "Total vect memory = %d Kbytes", Tot_mem / 1000);
+#endif
+
+    return (top);
+}
+
+/*! 
+   \brief Tracking memory 
+
+   \param minus mimus number 
+ */
+void sub_Vectmem(int minus)
+{
+    G_debug(5, "sub_Vectmem(): minus=%d", minus);
+#ifdef TRAK_MEM
+    {
+	Tot_mem -= minus;
+    }
+#endif
+
+    return;
+}
+
+/*!
+  \brief Load styles for geolines based on thematic mapping
+
+  \param gv pointer to geovect structure
+  \param colors pointer to Colors structure or NULL
+
+  \return number of features defined by thematic mapping
+  \return -1 on error
+*/
+int Gv_load_vect_thematic(geovect *gv, struct Colors *colors)
+{
+    geoline *gvt;
+
+    struct Map_info Map;
+    struct field_info *Fi;
+    
+    int nvals, cat, nlines, nskipped;
+    int red, blu, grn;
+    const char *str;
+    const char *mapset;
+
+    dbDriver *driver;
+    dbValue value;
+    
+    if(!gv || !gv->tstyle || !gv->filename)
+	return -1;
+
+    mapset = G_find_vector2(gv->filename, "");
+    if (!mapset) {
+	G_fatal_error(_("Vector map <%s> not found"), gv->filename);
+    }
+    
+    Vect_set_open_level(1);
+    if (Vect_open_old(&Map, gv->filename, "") == -1) {
+	G_fatal_error(_("Unable to open vector map <%s>"),
+		      G_fully_qualified_name(gv->filename, mapset));
+    }
+    
+    Fi = Vect_get_field(&Map, gv->tstyle->layer);
+    if (!Fi) {
+	G_warning(_("Database connection not defined for layer %d"),
+		  gv->tstyle->layer);
+    }
+    else {
+      driver = db_start_driver_open_database(Fi->driver, Fi->database);
+      if (!driver)
+	  G_fatal_error(_("Unable to open database <%s> by driver <%s>"),
+			Fi->database, Fi->driver);
+    }
+    G_message(_("Loading thematic vector layer <%s>..."),
+	      G_fully_qualified_name(gv->filename, mapset));
+    nlines = nskipped = 0;
+    for(gvt = gv->lines; gvt; gvt = gvt->next) {
+	gvt->style = (gvstyle *) G_malloc(sizeof(gvstyle));
+	G_zero(gvt->style, sizeof(gvstyle));
+	
+	/* use default style */
+	gvt->style->color  = gv->style->color;
+	gvt->style->symbol = gv->style->symbol;
+	gvt->style->size   = gv->style->size;
+	gvt->style->width  = gv->style->width;
+
+	cat = -1;
+	if (gvt->cats)
+	    Vect_cat_get(gvt->cats, gv->tstyle->layer, &cat);
+	if (cat < 0) {
+	    nskipped++;
+	    continue;
+	}
+	
+	/* color */
+	if (colors) {
+	    if (!Rast_get_c_color((const CELL *) &cat, &red, &grn, &blu, colors)) {
+		G_warning(_("No color rule defined for category %d"), cat);
+		gvt->style->color = gv->style->color;
+	    }
+	    gvt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
+		((int)((blu) << 16) & BLU_MASK);
+	}
+	
+	if (gv->tstyle->color_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gv->tstyle->color_column, &value);
+	    if (nvals < 1)
+		continue;
+	    str = db_get_value_string(&value);
+	    if (!str)
+		continue;
+	    if (G_str_to_color(str, &red, &grn, &blu) != 1) {
+		G_warning(_("Invalid color definition (%s)"),
+			  str);
+		gvt->style->color = gv->style->color;
+	    }
+	    else {
+		gvt->style->color = (red & RED_MASK) + ((int)((grn) << 8) & GRN_MASK) +
+		    ((int)((blu) << 16) & BLU_MASK);
+	    }
+	}
+	
+	/* width */
+	if (gv->tstyle->width_column) {
+	    nvals = db_select_value(driver, Fi->table, Fi->key, cat, gv->tstyle->width_column, &value);
+	    if (nvals < 1)
+		continue;
+	    gvt->style->width = db_get_value_int(&value);
+	}
+
+	nlines++;
+    }
+
+    if (nskipped > 0)
+	G_warning(_("%d features without category. "
+		    "Unable to determine color rules for features without category."),
+		  nskipped);
+    
+    return nlines;
+}

Copied: grass/trunk/lib/ogsf/gvl2.c (from rev 62429, grass/trunk/lib/ogsf/GVL2.c)
===================================================================
--- grass/trunk/lib/ogsf/gvl2.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gvl2.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,1521 @@
+/*!
+   \file GVL2.c
+
+   \brief OGSF library - loading and manipulating volumes
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Bill Brown UI-GMSL (May 1997)
+   Tomas Paudits (February 2004)
+ */
+
+#include <string.h>
+#include <grass/gis.h>
+#include <grass/raster3d.h>
+#include <grass/ogsf.h>
+#include <grass/glocale.h>
+#include "gsget.h"
+
+static int Vol_ID[MAX_VOLS];
+static int Next_vol = 0;
+
+static RASTER3D_Region wind3;
+static double Region[6];
+
+/*!
+   \brief Library intialization for volumes
+
+   Set region extent (N,S,W,E,T,B)
+ */
+void GVL_libinit(void)
+{
+    Rast3d_init_defaults();
+    Rast3d_get_window(&wind3);
+
+    Region[0] = wind3.north;
+    Region[1] = wind3.south;
+    Region[2] = wind3.west;
+    Region[3] = wind3.east;
+    Region[4] = wind3.top;
+    Region[5] = wind3.bottom;
+
+    return;
+}
+
+/*!
+   \brief Initialize 3D region
+
+   Set region extent (N,S,W,E,T,B)
+ */
+void GVL_init_region(void)
+{
+    Rast3d_read_window(&wind3, NULL);
+
+    Region[0] = wind3.north;
+    Region[1] = wind3.south;
+    Region[2] = wind3.west;
+    Region[3] = wind3.east;
+    Region[4] = wind3.top;
+    Region[5] = wind3.bottom;
+
+    return;
+}
+
+/*!
+   \brief Get region extent settings
+
+   \param[out] n,s,w,e north, south, west, east
+   \param[out] t,b top, bottom
+
+   \return 1
+ */
+int GVL_get_region(float *n, float *s, float *w, float *e, float *t, float *b)
+{
+    *n = Region[0];
+    *s = Region[1];
+    *w = Region[2];
+    *e = Region[3];
+    *t = Region[4];
+    *b = Region[5];
+
+    return (1);
+}
+
+/*!
+   \brief Get window 
+
+   \todo gvl_file.c use this - change
+
+   \return pointer to RASTER3D_Region struct (static)
+ */
+void *GVL_get_window()
+{
+    return &wind3;
+}
+
+/*!
+   \brief Check if volume set exists
+
+   \param id volume set id
+
+   \return 1 found
+   \return 0 not found
+ */
+int GVL_vol_exists(int id)
+{
+    int i, found = 0;
+
+    G_debug(3, "GVL_vol_exists");
+
+    if (NULL == gvl_get_vol(id)) {
+	return (0);
+    }
+
+    for (i = 0; i < Next_vol && !found; i++) {
+	if (Vol_ID[i] == id) {
+	    found = 1;
+	}
+    }
+
+    return (found);
+}
+
+/*!
+   \brief Create new volume set
+
+   \return volume set id
+   \return -1 on error
+ */
+int GVL_new_vol(void)
+{
+    geovol *nvl;
+
+    G_debug(3, "GVL_new_vol():");
+
+    if (Next_vol < MAX_VOLS) {
+	nvl = gvl_get_new_vol();
+
+	gvl_init_vol(nvl, wind3.west + wind3.ew_res / 2.,
+		     wind3.south + wind3.ns_res / 2., wind3.bottom,
+		     wind3.rows, wind3.cols, wind3.depths,
+		     wind3.ew_res, wind3.ns_res, wind3.tb_res);
+
+	Vol_ID[Next_vol] = nvl->gvol_id;
+	++Next_vol;
+
+	G_debug(3, "    id=%d", nvl->gvol_id);
+	
+	return (nvl->gvol_id);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get number of loaded volume sets
+
+   \return number of volume sets
+ */
+int GVL_num_vols(void)
+{
+    return (gvl_num_vols());
+}
+
+/*!
+   \brief Get list of loaded volume sets
+
+   Must be freed if not needed!
+
+   \param[out] numvols number of volume sets
+
+   \return pointer to list of volume sets
+   \return NULL on error
+ */
+int *GVL_get_vol_list(int *numvols)
+{
+    int i, *ret;
+
+    *numvols = Next_vol;
+
+    if (Next_vol) {
+	ret = (int *)G_malloc(Next_vol * sizeof(int));
+	if (!ret)
+	    return (NULL);
+
+	for (i = 0; i < Next_vol; i++) {
+	    ret[i] = Vol_ID[i];
+	}
+
+	return (ret);
+    }
+
+    return (NULL);
+}
+
+/*!
+   \brief Delete volume set from list
+
+   \param id volume set id
+
+   \return 1 on success
+   \return -1 on error (invalid volume set id)
+ */
+int GVL_delete_vol(int id)
+{
+    int i, j, found = 0;
+
+    G_debug(3, "GVL_delete_vol");
+
+    if (GVL_vol_exists(id)) {
+
+	for (i = 0; i < GVL_isosurf_num_isosurfs(id); i++) {
+	    GVL_isosurf_del(id, 0);
+	}
+
+	for (i = 0; i < GVL_slice_num_slices(id); i++) {
+	    GVL_slice_del(id, 0);
+	}
+
+	gvl_delete_vol(id);
+
+	for (i = 0; i < Next_vol && !found; i++) {
+	    if (Vol_ID[i] == id) {
+		found = 1;
+		for (j = i; j < Next_vol; j++) {
+		    Vol_ID[j] = Vol_ID[j + 1];
+		}
+	    }
+	}
+
+	if (found) {
+	    --Next_vol;
+
+	    return (1);
+	}
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Load 3d raster map to volume set
+
+   \param id volume set id
+   \param filename 3d raster map name
+
+   \return -1 on error
+   \return 0 on success
+ */
+int GVL_load_vol(int id, const char *filename)
+{
+    geovol *gvl;
+    int handle;
+
+    G_debug(3, "GVL_load_vol(): id=%d, name=%s", id, filename);
+
+    if (NULL == (gvl = gvl_get_vol(id))) {
+	return (-1);
+    }
+
+    G_message(_("Loading 3d raster map <%s>..."), filename);
+
+    if (0 > (handle = gvl_file_newh(filename, VOL_FTYPE_RASTER3D)))
+	return (-1);
+
+    gvl->hfile = handle;
+
+    return (0);
+}
+
+/*!
+   \brief Get volume set name
+
+   \param id volume set id
+   \param[out] filename name (must be allocated)
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_get_volname(int id, char *filename)
+{
+    geovol *gvl;
+
+    if (NULL == (gvl = gvl_get_vol(id))) {
+	return (-1);
+    }
+
+    if (0 > gvl->hfile) {
+	return (-1);
+    }
+
+    strcpy(filename, gvl_file_get_name(gvl->hfile));
+
+    return (1);
+}
+
+/*!
+   \brief Get volume dimensions
+
+   \param id volume set id
+   \param[out] rows,cols,depths number of rows, cols, depths
+ */
+void GVL_get_dims(int id, int *rows, int *cols, int *depths)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*rows = gvl->rows;
+	*cols = gvl->cols;
+	*depths = gvl->depths;
+    }
+
+    G_debug(3, "GVL_get_dims() id=%d, rows=%d, cols=%d, depths=%d",
+	    gvl->gvol_id, gvl->rows, gvl->cols, gvl->depths);
+    
+    return;
+}
+
+/*!
+   \brief Set trans ?
+
+   \param id volume set id
+   \param xtrans,ytrans,ztrans x/y/z trans values
+ */
+void GVL_set_trans(int id, float xtrans, float ytrans, float ztrans)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_set_trans");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->x_trans = xtrans;
+	gvl->y_trans = ytrans;
+	gvl->z_trans = ztrans;
+    }
+
+    return;
+}
+
+/*!
+   \brief Get trans ?
+
+   \param id volume set id
+   \param[out] xtrans,ytrans,ztrans x/y/z trans values
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_get_trans(int id, float *xtrans, float *ytrans, float *ztrans)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*xtrans = gvl->x_trans;
+	*ytrans = gvl->y_trans;
+	*ztrans = gvl->z_trans;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set drawing wire box
+
+   \param id volume set id
+   \param draw_wire 1 for drawing wire, 0 otherwise
+ */
+void GVL_set_draw_wire(int id, int draw_wire)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_set_draw_wire");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->draw_wire = draw_wire;
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw volume set
+
+   \param vid volume set id
+ */
+void GVL_draw_vol(int vid)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(vid);
+
+    if (gvl) {
+	gvld_vol(gvl);
+        if (gvl->draw_wire) {
+	    gvld_wind3_box(gvl);
+        }
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw volume in wire mode
+
+   \param id volume set id
+ */
+void GVL_draw_wire(int id)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_draw_wire(): id=%d", id);
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvld_wire_vol(gvl);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all volume sets
+ */
+void GVL_alldraw_vol(void)
+{
+    int id;
+
+    for (id = 0; id < Next_vol; id++) {
+	GVL_draw_vol(Vol_ID[id]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Draw all volume sets in wire mode
+ */
+void GVL_alldraw_wire(void)
+{
+    int id;
+
+    for (id = 0; id < Next_vol; id++) {
+	GVL_draw_wire(Vol_ID[id]);
+    }
+
+    return;
+}
+
+/*!
+   \brief Set client data for volume set
+
+   \param id volume set id
+   \param clientd pointer to client data
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_Set_ClientData(int id, void *clientd)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->clientdata = clientd;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get client data
+
+   \param id volume set id
+
+   \return pointer to client data
+   \return NULL on error
+ */
+void *GVL_Get_ClientData(int id)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	return (gvl->clientdata);
+    }
+
+    return (NULL);
+}
+
+/*!
+   \brief Set focus on map center
+
+   \param id volume set id
+ */
+void GVL_set_focus_center_map(int id)
+{
+    float center[3];
+    geovol *gvl;
+
+    G_debug(3, "GS_set_focus_center_map");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	center[X] = (gvl->xmax - gvl->xmin) / 2.;
+	center[Y] = (gvl->ymax - gvl->ymin) / 2.;
+	center[Z] = (gvl->zmax - gvl->zmin) / 2.;
+
+	GS_set_focus(center);
+    }
+
+    return;
+}
+
+/************************************************************************/
+/* ISOSURFACES */
+
+/************************************************************************/
+
+/*!
+   \brief Get draw resolution for isosurface
+
+   \todo error handling
+
+   \param id volume set id
+   \param[out] xres,yres,zres x/y/z resolution value
+ */
+void GVL_isosurf_get_drawres(int id, int *xres, int *yres, int *zres)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_isosurf_get_drawres");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*xres = gvl->isosurf_x_mod;
+	*yres = gvl->isosurf_y_mod;
+	*zres = gvl->isosurf_z_mod;
+    }
+
+    return;
+}
+
+/*!
+   \brief Set isosurface draw resolution
+
+   \param id volume set id
+   \param xres,yres,zres x/y/z resolution value
+
+   \return -1 on error (invalid values/volume set id)
+   \return 0 on success
+ */
+int GVL_isosurf_set_drawres(int id, int xres, int yres, int zres)
+{
+    geovol *gvl;
+    int i;
+
+    G_debug(3, "GVL_isosurf_set_drawres(): id=%d", id);
+
+    if (xres < 1 || yres < 1 || zres < 1) {
+	return (-1);
+    }
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->isosurf_x_mod = xres;
+	gvl->isosurf_y_mod = yres;
+	gvl->isosurf_z_mod = zres;
+
+	for (i = 0; i < gvl->n_isosurfs; i++) {
+	    gvl_isosurf_set_att_changed(gvl->isosurf[i], ATT_TOPO);
+	}
+
+	return (0);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get isosurface draw mode
+
+   \param id volume set id
+   \param[out] mode draw-mode
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_get_drawmode(int id, int *mode)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*mode = gvl->isosurf_draw_mode;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set isosurface draw mode
+
+   \param id volume set id
+   \param mode draw mode
+
+   \return 0 on success
+   \return -1 on error (invalid volume set id)
+ */
+int GVL_isosurf_set_drawmode(int id, int mode)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_isosurf_set_drawmode(): id=%d mode=%d", id, mode);
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->isosurf_draw_mode = mode;
+
+	return (0);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Add isosurface
+
+   \param id volume set id
+
+   \return -1 on error (invalid volume set id
+   \return 1 on success
+ */
+int GVL_isosurf_add(int id)
+{
+    geovol *gvl;
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_add() id=%d", id);
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (gvl->n_isosurfs == MAX_ISOSURFS)
+	return (-1);
+
+    isosurf = (geovol_isosurf *) G_malloc(sizeof(geovol_isosurf));
+    if (!isosurf) {
+	return (-1);
+    }
+
+    gvl_isosurf_init(isosurf);
+
+    gvl->n_isosurfs++;
+    gvl->isosurf[gvl->n_isosurfs - 1] = (geovol_isosurf *) isosurf;
+
+    return (1);
+}
+
+/*!
+   \brief Delete isosurface
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_isosurf_del(int id, int isosurf_id)
+{
+    geovol *gvl;
+    geovol_isosurf *isosurf;
+    int i;
+
+    G_debug(3, "GVL_isosurf_del");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (!isosurf)
+	return (-1);
+
+    if (!gvl_isosurf_freemem(isosurf)) {
+	return (-1);
+    }
+
+    gvl = gvl_get_vol(id);
+
+    G_free(gvl->isosurf[isosurf_id]);
+
+    for (i = isosurf_id + 1; i < gvl->n_isosurfs; i++) {
+	gvl->isosurf[i - 1] = gvl->isosurf[i];
+    }
+
+    gvl->n_isosurfs--;
+
+    return (1);
+}
+
+/*!
+   \brief Move up isosurface in list
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_isosurf_move_up(int id, int isosurf_id)
+{
+    geovol *gvl;
+    geovol_isosurf *tmp;
+
+    G_debug(3, "GVL_isosurf_move_up");
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (isosurf_id < 0 || isosurf_id > (gvl->n_isosurfs - 1))
+	return (-1);
+
+    if (isosurf_id == 0)
+	return (1);
+
+    tmp = gvl->isosurf[isosurf_id - 1];
+    gvl->isosurf[isosurf_id - 1] = gvl->isosurf[isosurf_id];
+    gvl->isosurf[isosurf_id] = tmp;
+
+    return (1);
+}
+
+/*!
+   \brief Move down isosurface in list
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_isosurf_move_down(int id, int isosurf_id)
+{
+    geovol *gvl;
+    geovol_isosurf *tmp;
+
+    G_debug(3, "GVL_isosurf_move_up");
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (isosurf_id < 0 || isosurf_id > (gvl->n_isosurfs - 1))
+	return (-1);
+
+    if (isosurf_id == (gvl->n_isosurfs - 1))
+	return (1);
+
+    tmp = gvl->isosurf[isosurf_id + 1];
+    gvl->isosurf[isosurf_id + 1] = gvl->isosurf[isosurf_id];
+    gvl->isosurf[isosurf_id] = tmp;
+
+    return (1);
+}
+
+/*!
+   \brief Get isosurface attributes
+
+   \param id volume set id
+   \param isosurf_id surface id
+   \param att attribute id
+   \param[out] set
+   \param[out] constant
+   \param[out] mapname
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_isosurf_get_att(int id, int isosurf_id,
+			int att, int *set, float *constant, char *mapname)
+{
+    int src;
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_get_att");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	if (-1 != (src = gvl_isosurf_get_att_src(isosurf, att))) {
+	    *set = src;
+
+	    if (src == CONST_ATT) {
+		*constant = isosurf->att[att].constant;
+	    }
+	    else if (src == MAP_ATT) {
+		strcpy(mapname, gvl_file_get_name(isosurf->att[att].hfile));
+	    }
+
+	    return (1);
+	}
+
+	return (-1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Unset isosurface attributes
+
+   \param id volume set id
+   \param isosurface_id isosurface id
+   \param att attribute id
+
+   \return ?
+   \return -1 on error
+ */
+int GVL_isosurf_unset_att(int id, int isosurf_id, int att)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_unset_att");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	return (gvl_isosurf_set_att_src(isosurf, att, NOTSET_ATT));
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set constant isosurface attribute
+
+   Attributes:
+    - ATT_NORM
+    - ATT_TOPO topography (level) constant
+    - ATT_COLOR color map/constant
+    - ATT_MASK mask map
+    - ATT_TRANSP transparency map/constant
+    - ATT_SHINE shininess map/constant
+    - ATT_EMIT emission map/constant
+
+   \param id volume set id
+   \param isosurf_id isosurface id (0 - MAX_ISOSURFS)
+   \param att attribute descriptor
+   \param constant constant value
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_set_att_const(int id, int isosurf_id, int att, float constant)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_set_att_const() id=%d isosurf_id=%d "
+	    "att=%d const=%f", id, isosurf_id, att, constant);
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	return (gvl_isosurf_set_att_const(isosurf, att, constant));
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set isosurface map attribute
+
+   Attributes:
+    - ATT_NORM
+    - ATT_TOPO topography (level) constant
+    - ATT_COLOR color map/constant
+    - ATT_MASK mask map
+    - ATT_TRANSP transparency map/constant
+    - ATT_SHINE shininess map/constant
+    - ATT_EMIT emission map/constant
+
+   \param id volume set id
+   \param isosurf_id isosurface id (0 - MAX_ISOSURFS)
+   \param att attribute descriptor
+   \param filename map name
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_set_att_map(int id, int isosurf_id, int att,
+			    const char *filename)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_set_att_map(): id=%d, isosurf_id=%d "
+	    "att=%d map=%s", id, isosurf_id, att, filename);
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	return gvl_isosurf_set_att_map(isosurf, att, filename);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get isosurface flags
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+   \param[out] inout map name
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_get_flags(int id, int isosurf_id, int *inout)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_get_flags");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	*inout = isosurf->inout_mode;
+
+	return (1);
+    }
+    return (-1);
+}
+
+/*!
+   \brief Set isosurface flags
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+   \param inout map name
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_set_flags(int id, int isosurf_id, int inout)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_get_flags");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	isosurf->inout_mode = inout;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get number of available isosurfaces
+
+   \param id volume set id
+
+   \return number of isosurfaces
+   \return -1 on error
+ */
+int GVL_isosurf_num_isosurfs(int id)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_isosurf_num_isosurfs");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	return gvl->n_isosurfs;
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set mask attribute mode
+
+   Mask attribute special: constant is set to indicate invert or no
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+   \param mode attribute mode
+
+   \return mode id
+   \return -1 on error
+ */
+int GVL_isosurf_set_maskmode(int id, int isosurf_id, int mode)
+{
+    geovol_isosurf *isosurf;
+
+    G_debug(3, "GVL_isosurf_set_att_const");
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	isosurf->att[ATT_MASK].constant = mode;
+
+	return (mode);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get isosurface mask mode
+
+   \param id volume set id
+   \param isosurf_id isosurface id
+   \param mode attribute mode
+
+   \return 1 on success
+   \return -1 on error
+ */
+int GVL_isosurf_get_maskmode(int id, int isosurf_id, int *mode)
+{
+    geovol_isosurf *isosurf;
+
+    isosurf = gvl_isosurf_get_isosurf(id, isosurf_id);
+
+    if (isosurf) {
+	*mode = isosurf->att[ATT_MASK].constant;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/************************************************************************/
+/* SLICES */
+
+/************************************************************************/
+
+/*!
+   \brief Get draw resolution of slice
+
+   \param id volume set id
+   \param[out] xres,yres,zres x/y/z resolution value
+ */
+void GVL_slice_get_drawres(int id, int *xres, int *yres, int *zres)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_slice_get_drawres");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*xres = gvl->slice_x_mod;
+	*yres = gvl->slice_y_mod;
+	*zres = gvl->slice_z_mod;
+    }
+
+    return;
+}
+
+/*!
+   \brief Set slice draw resolution
+
+   \param id volume set id
+   \param xres,yres,zres x/y/z resolution value
+
+   \return 0 on success
+   \return -1 on error (invalid value or id)
+ */
+int GVL_slice_set_drawres(int id, int xres, int yres, int zres)
+{
+    geovol *gvl;
+    int i;
+
+    G_debug(3, "GVL_slice_set_drawres(): id=%d", id);
+
+    if (xres < 1 || yres < 1 || zres < 1) {
+	return (-1);
+    }
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->slice_x_mod = xres;
+	gvl->slice_y_mod = yres;
+	gvl->slice_z_mod = zres;
+
+	for (i = 0; i < gvl->n_slices; i++) {
+	    gvl->slice[i]->changed = 1;
+	}
+
+	return (0);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get slice draw mode
+
+   \param id volume set id
+   \param[out] mode draw mode
+
+   \return 1 on success
+   \return -1 on error (invalid id)
+ */
+int GVL_slice_get_drawmode(int id, int *mode)
+{
+    geovol *gvl;
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	*mode = gvl->slice_draw_mode;
+
+	return (1);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Set slice draw mode
+
+   \param id volume set id
+   \param mode draw mode
+
+   \return 0 on success
+   \return -1 on error (invalid id)
+ */
+int GVL_slice_set_drawmode(int id, int mode)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_slice_set_drawmode(): id=%d, mode=%d", id, mode);
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	gvl->slice_draw_mode = mode;
+
+	return (0);
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Add slice
+
+   \param id volume set id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_add(int id)
+{
+    geovol *gvl;
+    geovol_slice *slice;
+
+    G_debug(3, "GVL_slice_add");
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (gvl->n_slices == MAX_SLICES)
+	return (-1);
+
+    if (NULL == (slice = (geovol_slice *) G_malloc(sizeof(geovol_slice)))) {
+	return (-1);
+    }
+
+    gvl_slice_init(slice);
+
+    gvl->n_slices++;
+    gvl->slice[gvl->n_slices - 1] = (geovol_slice *) slice;
+
+    return (1);
+}
+
+/*!
+   \brief Delete slice
+
+   \param id volume set id
+   \param slice_id slice id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_del(int id, int slice_id)
+{
+    geovol *gvl;
+    geovol_slice *slice;
+    int i;
+
+    G_debug(3, "GVL_slice_del");
+
+    slice = gvl_slice_get_slice(id, slice_id);
+
+    if (!slice)
+	return (-1);
+
+    if (!gvl_slice_freemem(slice)) {
+	return (-1);
+    }
+
+    gvl = gvl_get_vol(id);
+
+    G_free(gvl->slice[slice_id]);
+
+    for (i = slice_id + 1; i < gvl->n_slices; i++) {
+	gvl->slice[i - 1] = gvl->slice[i];
+    }
+
+    gvl->n_slices--;
+
+    return (1);
+}
+
+/*!
+   \brief Move up slice
+
+   \param id volume set id
+   \param slice_id slice id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_move_up(int id, int slice_id)
+{
+    geovol *gvl;
+    geovol_slice *tmp;
+
+    G_debug(3, "GVL_slice_move_up");
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (slice_id < 0 || slice_id > (gvl->n_slices - 1))
+	return (-1);
+
+    if (slice_id == 0)
+	return (1);
+
+    tmp = gvl->slice[slice_id - 1];
+    gvl->slice[slice_id - 1] = gvl->slice[slice_id];
+    gvl->slice[slice_id] = tmp;
+
+    return (1);
+}
+
+/*!
+   \brief Move down slice
+
+   \param id volume set id
+   \param slice_id slice id
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_move_down(int id, int slice_id)
+{
+    geovol *gvl;
+    geovol_slice *tmp;
+
+    G_debug(3, "GVL_slice_move_up");
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    if (slice_id < 0 || slice_id > (gvl->n_slices - 1))
+	return (-1);
+
+    if (slice_id == (gvl->n_slices - 1))
+	return (1);
+
+    tmp = gvl->slice[slice_id + 1];
+    gvl->slice[slice_id + 1] = gvl->slice[slice_id];
+    gvl->slice[slice_id] = tmp;
+
+    return (1);
+}
+
+/*!
+   \brief Get number or slices
+
+   \param id volume set id
+
+   \return number of slices
+   \return -1 on error
+ */
+int GVL_slice_num_slices(int id)
+{
+    geovol *gvl;
+
+    G_debug(3, "GVL_isosurf_num_isosurfs");
+
+    gvl = gvl_get_vol(id);
+
+    if (gvl) {
+	return gvl->n_slices;
+    }
+
+    return (-1);
+}
+
+/*!
+   \brief Get slice position
+
+   \param id volume set id
+   \param slice_id slice id
+   \param[out] x1,y1,z1 coordinates ?
+   \param[out] x2,y2,z2 coordinates ?
+   \param[out] dir direction
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_get_pos(int id, int slice_id,
+		      float *x1, float *x2, float *y1, float *y2, float *z1,
+		      float *z2, int *dir)
+{
+    geovol *gvl;
+    geovol_slice *slice;
+    int cols, rows, depths;
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    slice = gvl_slice_get_slice(id, slice_id);
+
+    if (!slice)
+	return (-1);
+
+    if (slice->dir == X) {
+	cols = gvl->rows;
+	rows = gvl->depths;
+	depths = gvl->cols;
+    }
+    else if (slice->dir == Y) {
+	cols = gvl->cols;
+	rows = gvl->depths;
+	depths = gvl->rows;
+    }
+    else if (slice->dir == Z) {
+	cols = gvl->cols;
+	rows = gvl->rows;
+	depths = gvl->depths;
+    }
+    else {
+	return (-1);
+    }
+
+    *x1 = slice->x1 / (cols - 1);
+    *x2 = slice->x2 / (cols - 1);
+    *y1 = slice->y1 / (rows - 1);
+    *y2 = slice->y2 / (rows - 1);
+    *z1 = slice->z1 / (depths - 1);
+    *z2 = slice->z2 / (depths - 1);
+
+    *dir = slice->dir;
+
+    return (1);
+}
+
+/*!
+   \brief Get slice position
+
+   \param id volume set id
+   \param slice_id slice id
+   \param x1,y1,z1 coordinates ?
+   \param x2,y2,z2 coordinates ?
+   \param dir direction
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_set_pos(int id, int slice_id,
+		      float x1, float x2, float y1, float y2, float z1,
+		      float z2, int dir)
+{
+    geovol *gvl;
+    geovol_slice *slice;
+    int cols, rows, depths;
+
+    gvl = gvl_get_vol(id);
+
+    if (!gvl)
+	return (-1);
+
+    slice = gvl_slice_get_slice(id, slice_id);
+
+    if (!slice)
+	return (-1);
+
+    if (dir == X) {
+	cols = gvl->rows;
+	rows = gvl->depths;
+	depths = gvl->cols;
+    }
+    else if (dir == Y) {
+	cols = gvl->cols;
+	rows = gvl->depths;
+	depths = gvl->rows;
+    }
+    else if (dir == Z) {
+	cols = gvl->cols;
+	rows = gvl->rows;
+	depths = gvl->depths;
+    }
+    else {
+	return (-1);
+    }
+
+    slice->x1 = ((x1 < 0.) ? 0. : ((x1 > 1.) ? 1. : x1)) * (cols - 1);
+    slice->x2 = ((x2 < 0.) ? 0. : ((x2 > 1.) ? 1. : x2)) * (cols - 1);
+    slice->y1 = ((y1 < 0.) ? 0. : ((y1 > 1.) ? 1. : y1)) * (rows - 1);
+    slice->y2 = ((y2 < 0.) ? 0. : ((y2 > 1.) ? 1. : y2)) * (rows - 1);
+    slice->z1 = ((z1 < 0.) ? 0. : ((z1 > 1.) ? 1. : z1)) * (depths - 1);
+    slice->z2 = ((z2 < 0.) ? 0. : ((z2 > 1.) ? 1. : z2)) * (depths - 1);
+
+    slice->dir = dir;
+
+    slice->changed = 1;
+
+    return (1);
+}
+
+/*!
+   \brief Get slice trans ?
+
+   \param id volume set id
+   \param slice_id slice id
+   \param[out] transp transp value
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_get_transp(int id, int slice_id, int *transp)
+{
+    geovol_slice *slice;
+
+    G_debug(3, "GVL_get_transp");
+
+    slice = gvl_slice_get_slice(id, slice_id);
+
+    if (!slice)
+	return (-1);
+
+    *transp = slice->transp;
+
+    return (1);
+}
+
+/*!
+   \brief Set slice trans ?
+
+   \param id volume set id
+   \param slice_id slice id
+   \param transp transp value
+
+   \return -1 on error
+   \return 1 on success
+ */
+int GVL_slice_set_transp(int id, int slice_id, int transp)
+{
+    geovol_slice *slice;
+
+    G_debug(3, "GVL_set_transp");
+
+    slice = gvl_slice_get_slice(id, slice_id);
+
+    if (!slice)
+	return (-1);
+
+    slice->transp = transp;
+
+    return (1);
+}

Copied: grass/trunk/lib/ogsf/gvl3.c (from rev 62429, grass/trunk/lib/ogsf/Gvl3.c)
===================================================================
--- grass/trunk/lib/ogsf/gvl3.c	                        (rev 0)
+++ grass/trunk/lib/ogsf/gvl3.c	2014-10-28 13:30:41 UTC (rev 62435)
@@ -0,0 +1,88 @@
+/*!
+   \file Gvl3.c
+
+   \brief OGSF library - loading volumes (lower level functions)
+
+   GRASS OpenGL gsurf OGSF Library 
+
+   (C) 1999-2008 by the GRASS Development Team
+
+   This program is free software under the 
+   GNU General Public License (>=v2). 
+   Read the file COPYING that comes with GRASS
+   for details.
+
+   \author Tomas Paudits (December 2003)
+   \author Doxygenized by Martin Landa <landa.martin gmail.com> (May 2008)
+ */
+
+#include <grass/gis.h>
+#include <grass/raster.h>
+#include <grass/raster3d.h>
+#include <grass/ogsf.h>
+#include <grass/glocale.h>
+
+/*!
+   \brief Load color table
+
+   \param[out] color_data color data buffer
+   \param name 3D raster map name
+
+   \return -1 on failure
+   \return 1 on success
+ */
+int Gvl_load_colors_data(void **color_data, const char *name)
+{
+    const char *mapset;
+    struct Colors *colors;
+
+    if (NULL == (mapset = G_find_raster3d(name, ""))) {
+	G_warning(_("3D raster map <%s> not found"), name);
+	return (-1);
+    }
+
+    if (NULL == (colors = (struct Colors *)G_malloc(sizeof(struct Colors))))
+	return (-1);
+
+    if (0 > Rast3d_read_colors(name, mapset, colors)) {
+	G_free(colors);
+	return (-1);
+    }
+
+    *color_data = colors;
+
+    return (1);
+}
+
+/*!
+   \brief Unload color table
+
+   \param color_data color data buffer
+
+   \return -1 on failure
+   \return 1 on success
+ */
+int Gvl_unload_colors_data(void *color_data)
+{
+    Rast_free_colors(color_data);
+
+    G_free(color_data);
+
+    return (1);
+}
+
+/*!
+   \brief Get color for value
+
+   \param color_data color data value
+   \param value data value
+
+   \return color value
+ */
+int Gvl_get_color_for_value(void *color_data, float *value)
+{
+    int r, g, b;
+
+    Rast_get_f_color((FCELL *) value, &r, &g, &b, color_data);
+    return ((r & 0xff) | ((g & 0xff) << 8) | ((b & 0xff) << 16));
+}



More information about the grass-commit mailing list