GRASS GIS 7 Programmer's Manual  7.9.dev(2021)-e5379bbd7
xnmode.c
Go to the documentation of this file.
1 
2 #include <stdlib.h>
3 
4 #include <grass/gis.h>
5 #include <grass/raster.h>
6 #include <grass/calc.h>
7 
8 /**********************************************************************
9 mode(x1,x2,..,xn)
10  return mode of arguments
11 **********************************************************************/
12 
13 static int dcmp(const void *aa, const void *bb)
14 {
15  const double *a = aa;
16  const double *b = bb;
17 
18  if (*a < *b)
19  return -1;
20  if (*a > *b)
21  return 1;
22  return 0;
23 }
24 
25 static double mode(double *value, int argc)
26 {
27  double mode_v;
28  int mode_n = 0;
29  int i;
30 
31  qsort(value, argc, sizeof(double), dcmp);
32 
33  for (i = 0; i < argc;) {
34  int n = 1;
35  double v = value[i];
36 
37  for (i++; i < argc; i++) {
38  if (value[i] != v)
39  break;
40  n++;
41  }
42 
43  if (n < mode_n)
44  continue;
45 
46  mode_v = v;
47  mode_n = n;
48  }
49 
50  return mode_v;
51 }
52 
53 int f_nmode(int argc, const int *argt, void **args)
54 {
55  static double *value;
56  static int value_size;
57  int size = argc * sizeof(double);
58  int i, j;
59 
60  if (argc < 1)
61  return E_ARG_LO;
62 
63  for (i = 1; i <= argc; i++)
64  if (argt[i] != argt[0])
65  return E_ARG_TYPE;
66 
67  if (size > value_size) {
68  value_size = size;
69  value = G_realloc(value, value_size);
70  }
71 
72  switch (argt[argc]) {
73  case CELL_TYPE:
74  {
75  CELL *res = args[0];
76  CELL **argv = (CELL **) & args[1];
77 
78  for (i = 0; i < columns; i++) {
79  int n = 0;
80 
81  for (j = 0; j < argc; j++) {
82  if (IS_NULL_C(&argv[j][i]))
83  continue;
84  value[n++] = (double)argv[j][i];
85  }
86 
87  if (!n)
88  SET_NULL_C(&res[i]);
89  else
90  res[i] = (CELL) mode(value, n);
91  }
92  return 0;
93  }
94  case FCELL_TYPE:
95  {
96  FCELL *res = args[0];
97  FCELL **argv = (FCELL **) & args[1];
98 
99  for (i = 0; i < columns; i++) {
100  int n = 0;
101 
102  for (j = 0; j < argc; j++) {
103  if (IS_NULL_F(&argv[j][i]))
104  continue;
105  value[n++] = (double)argv[j][i];
106  }
107 
108  if (!n)
109  SET_NULL_F(&res[i]);
110  else
111  res[i] = (FCELL) mode(value, n);
112  }
113  return 0;
114  }
115  case DCELL_TYPE:
116  {
117  DCELL *res = args[0];
118  DCELL **argv = (DCELL **) & args[1];
119 
120  for (i = 0; i < columns; i++) {
121  int n = 0;
122 
123  for (j = 0; j < argc; j++) {
124  if (IS_NULL_D(&argv[j][i]))
125  continue;
126  value[n++] = (double)argv[j][i];
127  }
128 
129  if (!n)
130  SET_NULL_D(&res[i]);
131  else
132  res[i] = (DCELL) mode(value, n);
133  }
134  return 0;
135  }
136  default:
137  return E_INV_TYPE;
138  }
139 }
#define CELL_TYPE
Definition: raster.h:11
#define SET_NULL_C(x)
Definition: calc.h:32
double DCELL
Definition: gis.h:603
int f_nmode(int argc, const int *argt, void **args)
Definition: xnmode.c:53
#define IS_NULL_F(x)
Definition: calc.h:29
#define IS_NULL_C(x)
Definition: calc.h:28
int columns
Definition: calc.c:12
#define DCELL_TYPE
Definition: raster.h:13
double b
Definition: r_raster.c:39
Definition: calc.h:12
float FCELL
Definition: gis.h:604
#define IS_NULL_D(x)
Definition: calc.h:30
int CELL
Definition: gis.h:602
#define G_realloc(p, n)
Definition: defs/gis.h:114
#define FCELL_TYPE
Definition: raster.h:12
#define SET_NULL_F(x)
Definition: calc.h:33
#define SET_NULL_D(x)
Definition: calc.h:34