1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 Sun Microsystems
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; version 2 of the License.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20
#include <drizzled/server_includes.h>
22
#include <drizzled/functions/min_max.h>
23
#include <drizzled/item/cmpfunc.h>
25
void Item_func_min_max::fix_length_and_dec()
28
bool datetime_found= false;
32
cmp_type=args[0]->result_type();
34
for (uint32_t i=0 ; i < arg_count ; i++)
36
set_if_bigger(max_length, args[i]->max_length);
37
set_if_bigger(decimals, args[i]->decimals);
38
set_if_bigger(max_int_part, args[i]->decimal_int_part());
39
if (args[i]->maybe_null)
41
cmp_type=item_cmp_type(cmp_type,args[i]->result_type());
42
if (args[i]->result_type() != ROW_RESULT && args[i]->is_datetime())
45
if (!datetime_item || args[i]->field_type() == DRIZZLE_TYPE_DATETIME)
46
datetime_item= args[i];
49
if (cmp_type == STRING_RESULT)
51
agg_arg_charsets(collation, args, arg_count, MY_COLL_CMP_CONV, 1);
54
session= current_session;
55
compare_as_dates= true;
58
else if ((cmp_type == DECIMAL_RESULT) || (cmp_type == INT_RESULT))
59
max_length= my_decimal_precision_to_length(max_int_part+decimals, decimals,
61
cached_field_type= agg_field_type(args, arg_count);
66
Compare item arguments in the DATETIME context.
70
value [out] found least/greatest DATE/DATETIME value
73
Compare item arguments as DATETIME values and return the index of the
74
least/greatest argument in the arguments array.
75
The correct integer DATE/DATETIME value of the found argument is
76
stored to the value pointer, if latter is provided.
79
0 If one of arguments is NULL
80
# index of the least/greatest argument
83
uint32_t Item_func_min_max::cmp_datetimes(uint64_t *value)
86
uint32_t min_max_idx= 0;
88
for (uint32_t i=0; i < arg_count ; i++)
92
uint64_t res= get_datetime_value(session, &arg, 0, datetime_item, &is_null);
93
if ((null_value= args[i]->null_value))
95
if (i == 0 || (res < min_max ? cmp_sign : -cmp_sign) > 0)
104
if (datetime_item->field_type() == DRIZZLE_TYPE_DATE)
111
String *Item_func_min_max::val_str(String *str)
114
if (compare_as_dates)
117
uint32_t min_max_idx= cmp_datetimes(NULL);
120
str_res= args[min_max_idx]->val_str(str);
121
str_res->set_charset(collation.collation);
127
int64_t nr=val_int();
130
str->set_int(nr, unsigned_flag, &my_charset_bin);
135
my_decimal dec_buf, *dec_val= val_decimal(&dec_buf);
138
my_decimal2string(E_DEC_FATAL_ERROR, dec_val, 0, 0, 0, str);
143
double nr= val_real();
145
return 0; /* purecov: inspected */
146
str->set_real(nr,decimals,&my_charset_bin);
153
for (uint32_t i=0; i < arg_count ; i++)
156
res=args[i]->val_str(str);
160
res2= args[i]->val_str(res == str ? &tmp_value : str);
163
int cmp= sortcmp(res,res2,collation.collation);
164
if ((cmp_sign < 0 ? cmp : -cmp) < 0)
168
if ((null_value= args[i]->null_value))
171
res->set_charset(collation.collation);
176
// This case should never be chosen
180
return 0; // Keep compiler happy
184
double Item_func_min_max::val_real()
188
if (compare_as_dates)
191
(void)cmp_datetimes(&result);
192
return (double)result;
194
for (uint32_t i=0; i < arg_count ; i++)
197
value= args[i]->val_real();
200
double tmp= args[i]->val_real();
201
if (!args[i]->null_value && (tmp < value ? cmp_sign : -cmp_sign) > 0)
204
if ((null_value= args[i]->null_value))
211
int64_t Item_func_min_max::val_int()
215
if (compare_as_dates)
218
(void)cmp_datetimes(&result);
219
return (int64_t)result;
221
for (uint32_t i=0; i < arg_count ; i++)
224
value=args[i]->val_int();
227
int64_t tmp=args[i]->val_int();
228
if (!args[i]->null_value && (tmp < value ? cmp_sign : -cmp_sign) > 0)
231
if ((null_value= args[i]->null_value))
238
my_decimal *Item_func_min_max::val_decimal(my_decimal *dec)
241
my_decimal tmp_buf, *tmp, *res= NULL;
243
if (compare_as_dates)
246
(void)cmp_datetimes(&value);
247
uint64_t2decimal(value, dec);
250
for (uint32_t i=0; i < arg_count ; i++)
253
res= args[i]->val_decimal(dec);
256
tmp= args[i]->val_decimal(&tmp_buf); // Zero if NULL
257
if (tmp && (my_decimal_cmp(tmp, res) * cmp_sign) < 0)
261
/* Move value out of tmp_buf as this will be reused on next loop */
262
my_decimal2decimal(tmp, dec);
269
if ((null_value= args[i]->null_value))