18
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21
#ifdef USE_PRAGMA_IMPLEMENTATION
22
#pragma implementation // gcc: Class implementation
29
25
#include <drizzled/field/double.h>
30
#include <drizzled/error.h>
31
#include <drizzled/table.h>
32
#include <drizzled/session.h>
33
#include "drizzled/internal/m_string.h"
40
27
/****************************************************************************
41
28
double precision floating point numbers
42
29
****************************************************************************/
44
int Field_double::store(const char *from,uint32_t len, const CHARSET_INFO * const cs)
31
int Field_double::store(const char *from,uint len,CHARSET_INFO *cs)
48
35
double nr= my_strntod(cs,(char*) from, len, &end, &error);
50
ASSERT_COLUMN_MARKED_FOR_WRITE;
51
if (error || (!len || (((uint32_t) (end-from) != len) && getTable()->in_use->count_cuted_fields)))
36
if (error || (!len || (((uint) (end-from) != len) && table->in_use->count_cuted_fields)))
53
set_warning(DRIZZLE_ERROR::WARN_LEVEL_WARN,
54
(error ? ER_WARN_DATA_OUT_OF_RANGE : ER_WARN_DATA_TRUNCATED), 1);
38
set_warning(MYSQL_ERROR::WARN_LEVEL_WARN,
39
(error ? ER_WARN_DATA_OUT_OF_RANGE : WARN_DATA_TRUNCATED), 1);
55
40
error= error ? 1 : 2;
57
42
Field_double::store(nr);
64
49
int error= truncate(&nr, DBL_MAX);
66
ASSERT_COLUMN_MARKED_FOR_WRITE;
68
51
#ifdef WORDS_BIGENDIAN
69
if (getTable()->getShare()->db_low_byte_first)
52
if (table->s->db_low_byte_first)
71
54
float8store(ptr,nr);
70
If a field has fixed length, truncate the double argument pointed to by 'nr'
72
Also ensure that the argument is within [-max_value; max_value] range.
75
int Field_real::truncate(double *nr, double max_value)
84
set_warning(MYSQL_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_OUT_OF_RANGE, 1);
87
else if (unsigned_flag && res < 0)
90
set_warning(MYSQL_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_OUT_OF_RANGE, 1);
96
uint order= field_length - dec;
97
uint step= array_elements(log_10) - 1;
99
for (; order > step; order-= step)
100
max_value*= log_10[step];
101
max_value*= log_10[order];
102
max_value-= 1.0 / log_10[dec];
104
/* Check for infinity so we don't get NaN in calculations */
107
double tmp= rint((res - floor(res)) * log_10[dec]) / log_10[dec];
108
res= floor(res) + tmp;
112
if (res < -max_value)
115
set_warning(MYSQL_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_OUT_OF_RANGE, 1);
117
else if (res > max_value)
120
set_warning(MYSQL_ERROR::WARN_LEVEL_WARN, ER_WARN_DATA_OUT_OF_RANGE, 1);
131
int Field_real::store_decimal(const my_decimal *dm)
134
my_decimal2double(E_DEC_FATAL_ERROR, dm, &dbl);
86
138
double Field_double::val_real(void)
90
ASSERT_COLUMN_MARKED_FOR_READ;
92
141
#ifdef WORDS_BIGENDIAN
93
if (getTable()->getShare()->db_low_byte_first)
142
if (table->s->db_low_byte_first)
133
179
char buf[DOUBLE_TO_STRING_CONVERSION_BUFFER_SIZE];
134
String tmp(buf, sizeof(buf), &my_charset_utf8_general_ci), *str;
135
str= val_str(&tmp, &tmp);
136
Session *session= getTable() ? getTable()->in_use : current_session;
137
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
180
String tmp(buf, sizeof(buf), &my_charset_latin1), *str;
181
str= val_str(&tmp, 0);
182
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_WARN,
138
183
ER_TRUNCATED_WRONG_VALUE,
139
184
ER(ER_TRUNCATED_WRONG_VALUE), "INTEGER",
191
my_decimal *Field_real::val_decimal(my_decimal *decimal_value)
193
double2my_decimal(E_DEC_FATAL_ERROR, val_real(), decimal_value);
194
return decimal_value;
146
198
String *Field_double::val_str(String *val_buffer,
199
String *val_ptr __attribute__((unused)))
151
ASSERT_COLUMN_MARKED_FOR_READ;
153
202
#ifdef WORDS_BIGENDIAN
154
if (getTable()->getShare()->db_low_byte_first)
203
if (table->s->db_low_byte_first)
156
205
float8get(nr,ptr);
160
209
doubleget(nr,ptr);
162
uint32_t to_length= max(field_length, (uint32_t)DOUBLE_TO_STRING_CONVERSION_BUFFER_SIZE);
211
uint to_length=max(field_length, DOUBLE_TO_STRING_CONVERSION_BUFFER_SIZE);
163
212
val_buffer->alloc(to_length);
164
213
char *to=(char*) val_buffer->ptr();
167
216
if (dec >= NOT_FIXED_DEC)
168
len= internal::my_gcvt(nr, internal::MY_GCVT_ARG_DOUBLE, to_length - 1, to, NULL);
217
len= my_gcvt(nr, MY_GCVT_ARG_DOUBLE, to_length - 1, to, NULL);
170
len= internal::my_fcvt(nr, dec, to, NULL);
219
len= my_fcvt(nr, dec, to, NULL);
172
val_buffer->length((uint32_t) len);
221
val_buffer->length((uint) len);
174
223
return val_buffer;
177
int Field_double::cmp(const unsigned char *a_ptr, const unsigned char *b_ptr)
226
bool Field_double::send_binary(Protocol *protocol)
228
return protocol->store((double) Field_double::val_real(), dec, (String*) 0);
232
int Field_double::cmp(const uchar *a_ptr, const uchar *b_ptr)
180
235
#ifdef WORDS_BIGENDIAN
181
if (getTable()->getShare()->db_low_byte_first)
236
if (table->s->db_low_byte_first)
183
238
float8get(a,a_ptr);
184
239
float8get(b,b_ptr);
251
#define DBL_EXP_DIG (sizeof(double)*8-DBL_MANT_DIG)
196
253
/* The following should work for IEEE */
198
void Field_double::sort_string(unsigned char *to,uint32_t )
255
void Field_double::sort_string(uchar *to,uint length __attribute__((unused)))
201
258
#ifdef WORDS_BIGENDIAN
202
if (getTable()->getShare()->db_low_byte_first)
259
if (table->s->db_low_byte_first)
204
261
float8get(nr,ptr);
271
Save the field metadata for double fields.
273
Saves the pack length in the first byte of the field metadata array
274
at index of *metadata_ptr.
276
@param metadata_ptr First byte of field metadata
278
@returns number of bytes written to metadata_ptr
280
int Field_double::do_save_field_metadata(uchar *metadata_ptr)
282
*metadata_ptr= pack_length();
213
287
void Field_double::sql_type(String &res) const
215
const CHARSET_INFO * const cs=res.charset();
289
CHARSET_INFO *cs=res.charset();
216
290
if (dec == NOT_FIXED_DEC)
218
292
res.set_ascii(STRING_WITH_LEN("double"));
222
296
res.length(cs->cset->snprintf(cs,(char*) res.ptr(),res.alloced_length(),
223
297
"double(%d,%d)",(int) field_length,dec));
227
} /* namespace drizzled */