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
22
#include <drizzled/function/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,
94
if ((null_value= args[i]->null_value))
96
if (i == 0 || (res < min_max ? cmp_sign : -cmp_sign) > 0)
105
if (datetime_item->field_type() == DRIZZLE_TYPE_DATE)
112
String *Item_func_min_max::val_str(String *str)
115
if (compare_as_dates)
118
uint32_t min_max_idx= cmp_datetimes(NULL);
121
str_res= args[min_max_idx]->val_str(str);
122
str_res->set_charset(collation.collation);
128
int64_t nr=val_int();
131
str->set_int(nr, unsigned_flag, &my_charset_bin);
136
my_decimal dec_buf, *dec_val= val_decimal(&dec_buf);
139
my_decimal2string(E_DEC_FATAL_ERROR, dec_val, 0, 0, 0, str);
144
double nr= val_real();
147
str->set_real(nr,decimals,&my_charset_bin);
154
for (uint32_t i=0; i < arg_count ; i++)
157
res=args[i]->val_str(str);
161
res2= args[i]->val_str(res == str ? &tmp_value : str);
164
int cmp= sortcmp(res,res2,collation.collation);
165
if ((cmp_sign < 0 ? cmp : -cmp) < 0)
169
if ((null_value= args[i]->null_value))
172
res->set_charset(collation.collation);
177
// This case should never be chosen
181
return 0; // Keep compiler happy
185
double Item_func_min_max::val_real()
189
if (compare_as_dates)
192
(void)cmp_datetimes(&result);
193
return (double)result;
195
for (uint32_t i=0; i < arg_count ; i++)
198
value= args[i]->val_real();
201
double tmp= args[i]->val_real();
202
if (!args[i]->null_value && (tmp < value ? cmp_sign : -cmp_sign) > 0)
205
if ((null_value= args[i]->null_value))
212
int64_t Item_func_min_max::val_int()
216
if (compare_as_dates)
219
(void)cmp_datetimes(&result);
220
return (int64_t)result;
222
for (uint32_t i=0; i < arg_count ; i++)
225
value=args[i]->val_int();
228
int64_t tmp=args[i]->val_int();
229
if (!args[i]->null_value && (tmp < value ? cmp_sign : -cmp_sign) > 0)
232
if ((null_value= args[i]->null_value))
239
my_decimal *Item_func_min_max::val_decimal(my_decimal *dec)
242
my_decimal tmp_buf, *tmp, *res= NULL;
244
if (compare_as_dates)
247
(void)cmp_datetimes(&value);
248
uint64_t2decimal(value, dec);
251
for (uint32_t i=0; i < arg_count ; i++)
254
res= args[i]->val_decimal(dec);
257
tmp= args[i]->val_decimal(&tmp_buf); // Zero if NULL
258
if (tmp && (my_decimal_cmp(tmp, res) * cmp_sign) < 0)
262
/* Move value out of tmp_buf as this will be reused on next loop */
263
my_decimal2decimal(tmp, dec);
270
if ((null_value= args[i]->null_value))