18
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
23
#include <drizzled/field/decimal.h>
24
#include <drizzled/error.h>
25
#include <drizzled/table.h>
26
#include <drizzled/session.h>
28
extern my_decimal decimal_zero;
21
#ifdef USE_PRAGMA_IMPLEMENTATION
22
#pragma implementation // gcc: Class implementation
25
#include <drizzled/field/fdecimal.h>
30
27
/****************************************************************************
32
****************************************************************************/
34
Field_decimal::Field_decimal(unsigned char *ptr_arg,
36
unsigned char *null_ptr_arg,
37
unsigned char null_bit_arg,
38
enum utype unireg_check_arg,
39
const char *field_name_arg,
52
precision= my_decimal_length_to_precision(len_arg, dec_arg, unsigned_arg);
53
set_if_smaller(precision, (uint32_t)DECIMAL_MAX_PRECISION);
54
assert((precision <= DECIMAL_MAX_PRECISION) &&
55
(dec <= DECIMAL_MAX_SCALE));
56
bin_size= my_decimal_get_binary_size(precision, dec);
59
Field_decimal::Field_decimal(uint32_t len_arg,
64
:Field_num((unsigned char*) 0,
66
maybe_null_arg ? (unsigned char*) "": 0,
74
precision= my_decimal_length_to_precision(len_arg, dec_arg, unsigned_arg);
75
set_if_smaller(precision, (uint32_t)DECIMAL_MAX_PRECISION);
76
assert((precision <= DECIMAL_MAX_PRECISION) &&
77
(dec <= DECIMAL_MAX_SCALE));
78
bin_size= my_decimal_get_binary_size(precision, dec);
82
int Field_decimal::reset(void)
29
****************************************************************************/
31
Field_new_decimal::Field_new_decimal(uchar *ptr_arg,
32
uint32_t len_arg, uchar *null_ptr_arg,
34
enum utype unireg_check_arg,
35
const char *field_name_arg,
36
uint8_t dec_arg,bool zero_arg,
38
:Field_num(ptr_arg, len_arg, null_ptr_arg, null_bit_arg,
39
unireg_check_arg, field_name_arg, dec_arg, zero_arg, unsigned_arg)
41
precision= my_decimal_length_to_precision(len_arg, dec_arg, unsigned_arg);
42
set_if_smaller(precision, DECIMAL_MAX_PRECISION);
43
assert((precision <= DECIMAL_MAX_PRECISION) &&
44
(dec <= DECIMAL_MAX_SCALE));
45
bin_size= my_decimal_get_binary_size(precision, dec);
49
Field_new_decimal::Field_new_decimal(uint32_t len_arg,
54
:Field_num((uchar*) 0, len_arg,
55
maybe_null_arg ? (uchar*) "": 0, 0,
56
NONE, name, dec_arg, 0, unsigned_arg)
58
precision= my_decimal_length_to_precision(len_arg, dec_arg, unsigned_arg);
59
set_if_smaller(precision, DECIMAL_MAX_PRECISION);
60
assert((precision <= DECIMAL_MAX_PRECISION) &&
61
(dec <= DECIMAL_MAX_SCALE));
62
bin_size= my_decimal_get_binary_size(precision, dec);
66
int Field_new_decimal::reset(void)
84
68
store_value(&decimal_zero);
249
int Field_decimal::store_decimal(const my_decimal *decimal_value)
238
int Field_new_decimal::store_decimal(const my_decimal *decimal_value)
251
240
return store_value(decimal_value);
255
int Field_decimal::store_time(DRIZZLE_TIME *ltime,
256
enum enum_drizzle_timestamp_type )
244
int Field_new_decimal::store_time(MYSQL_TIME *ltime,
245
timestamp_type t_type __attribute__((unused)))
258
my_decimal decimal_value;
259
return store_value(date2my_decimal(ltime, &decimal_value));
247
my_decimal decimal_value;
248
return store_value(date2my_decimal(ltime, &decimal_value));
263
double Field_decimal::val_real(void)
252
double Field_new_decimal::val_real(void)
266
255
my_decimal decimal_value;
268
ASSERT_COLUMN_MARKED_FOR_READ;
270
256
my_decimal2double(E_DEC_FATAL_ERROR, val_decimal(&decimal_value), &dbl);
276
int64_t Field_decimal::val_int(void)
261
int64_t Field_new_decimal::val_int(void)
279
264
my_decimal decimal_value;
281
ASSERT_COLUMN_MARKED_FOR_READ;
283
my_decimal2int(E_DEC_FATAL_ERROR, val_decimal(&decimal_value), false, &i);
265
my_decimal2int(E_DEC_FATAL_ERROR, val_decimal(&decimal_value),
289
my_decimal* Field_decimal::val_decimal(my_decimal *decimal_value)
271
my_decimal* Field_new_decimal::val_decimal(my_decimal *decimal_value)
291
ASSERT_COLUMN_MARKED_FOR_READ;
293
273
binary2my_decimal(E_DEC_FATAL_ERROR, ptr, decimal_value,
295
275
return(decimal_value);
299
String *Field_decimal::val_str(String *val_buffer,
279
String *Field_new_decimal::val_str(String *val_buffer,
280
String *val_ptr __attribute__((unused)))
302
282
my_decimal decimal_value;
304
ASSERT_COLUMN_MARKED_FOR_READ;
306
uint32_t fixed_precision= decimal_precision ? precision : 0;
283
uint fixed_precision= decimal_precision ? precision : 0;
307
284
my_decimal2string(E_DEC_FATAL_ERROR, val_decimal(&decimal_value),
308
285
fixed_precision, dec, '0', val_buffer);
309
286
return val_buffer;
313
int Field_decimal::cmp(const unsigned char *a,const unsigned char*b)
290
int Field_new_decimal::cmp(const uchar *a,const uchar*b)
315
292
return memcmp(a, b, bin_size);
319
void Field_decimal::sort_string(unsigned char *buff,
296
void Field_new_decimal::sort_string(uchar *buff,
297
uint length __attribute__((unused)))
322
299
memcpy(buff, ptr, bin_size);
326
void Field_decimal::sql_type(String &str) const
303
void Field_new_decimal::sql_type(String &str) const
328
const CHARSET_INFO * const cs= str.charset();
305
CHARSET_INFO *cs= str.charset();
329
306
str.length(cs->cset->snprintf(cs, (char*) str.ptr(), str.alloced_length(),
330
307
"decimal(%d,%d)", precision, (int)dec));
335
Returns the number of bytes field uses in row-based replication
338
This method is used in row-based replication to determine the number
339
of bytes that the field consumes in the row record format. This is
340
used to skip fields in the master that do not exist on the slave.
342
@param field_metadata Encoded size in field metadata
344
@returns The size of the field based on the field metadata.
346
uint32_t Field_decimal::pack_length_from_metadata(uint32_t field_metadata)
348
uint32_t const source_precision= (field_metadata >> 8U) & 0x00ff;
349
uint32_t const source_decimal= field_metadata & 0x00ff;
350
uint32_t const source_size= my_decimal_get_binary_size(source_precision,
313
Save the field metadata for new decimal fields.
315
Saves the precision in the first byte and decimals() in the second
316
byte of the field metadata array at index of *metadata_ptr and
319
@param metadata_ptr First byte of field metadata
321
@returns number of bytes written to metadata_ptr
323
int Field_new_decimal::do_save_field_metadata(uchar *metadata_ptr)
325
*metadata_ptr= precision;
326
*(metadata_ptr + 1)= decimals();
332
Returns the number of bytes field uses in row-based replication
335
This method is used in row-based replication to determine the number
336
of bytes that the field consumes in the row record format. This is
337
used to skip fields in the master that do not exist on the slave.
339
@param field_metadata Encoded size in field metadata
341
@returns The size of the field based on the field metadata.
343
uint Field_new_decimal::pack_length_from_metadata(uint field_metadata)
345
uint const source_precision= (field_metadata >> 8U) & 0x00ff;
346
uint const source_decimal= field_metadata & 0x00ff;
347
uint const source_size= my_decimal_get_binary_size(source_precision,
352
349
return (source_size);
357
Check to see if field size is compatible with destination.
359
This method is used in row-based replication to verify that the slave's
360
field size is less than or equal to the master's field size. The
361
encoded field metadata (from the master or source) is decoded and compared
362
to the size of this field (the slave or destination).
364
@param field_metadata Encoded size in field metadata
366
@retval 0 if this field's size is < the source field's size
367
@retval 1 if this field's size is >= the source field's size
354
Check to see if field size is compatible with destination.
356
This method is used in row-based replication to verify that the slave's
357
field size is less than or equal to the master's field size. The
358
encoded field metadata (from the master or source) is decoded and compared
359
to the size of this field (the slave or destination).
361
@param field_metadata Encoded size in field metadata
363
@retval 0 if this field's size is < the source field's size
364
@retval 1 if this field's size is >= the source field's size
369
int Field_decimal::compatible_field_size(uint32_t field_metadata)
366
int Field_new_decimal::compatible_field_size(uint field_metadata)
371
368
int compatible= 0;
372
uint32_t const source_precision= (field_metadata >> 8U) & 0x00ff;
373
uint32_t const source_decimal= field_metadata & 0x00ff;
374
uint32_t const source_size= my_decimal_get_binary_size(source_precision,
376
uint32_t const destination_size= row_pack_length();
369
uint const source_precision= (field_metadata >> 8U) & 0x00ff;
370
uint const source_decimal= field_metadata & 0x00ff;
371
uint const source_size= my_decimal_get_binary_size(source_precision,
373
uint const destination_size= row_pack_length();
377
374
compatible= (source_size <= destination_size);
379
376
compatible= (source_precision <= precision) &&
380
(source_decimal <= decimals());
377
(source_decimal <= decimals());
381
378
return (compatible);
385
uint32_t Field_decimal::is_equal(CreateField *new_field_ptr)
382
uint Field_new_decimal::is_equal(Create_field *new_field)
387
return ((new_field_ptr->sql_type == real_type()) &&
388
((new_field_ptr->flags & UNSIGNED_FLAG) ==
389
(uint32_t) (flags & UNSIGNED_FLAG)) &&
390
((new_field_ptr->flags & AUTO_INCREMENT_FLAG) ==
391
(uint32_t) (flags & AUTO_INCREMENT_FLAG)) &&
392
(new_field_ptr->length == max_display_length()) &&
393
(new_field_ptr->decimals == dec));
384
return ((new_field->sql_type == real_type()) &&
385
((new_field->flags & UNSIGNED_FLAG) ==
386
(uint) (flags & UNSIGNED_FLAG)) &&
387
((new_field->flags & AUTO_INCREMENT_FLAG) ==
388
(uint) (flags & AUTO_INCREMENT_FLAG)) &&
389
(new_field->length == max_display_length()) &&
390
(new_field->decimals == dec));
398
Unpack a decimal field from row data.
400
This method is used to unpack a decimal or numeric field from a master
401
whose size of the field is less than that of the slave.
403
@param to Destination of the data
404
@param from Source of the data
405
@param param_data Precision (upper) and decimal (lower) values
407
@return New pointer into memory based on from + length of the data
395
Unpack a decimal field from row data.
397
This method is used to unpack a decimal or numeric field from a master
398
whose size of the field is less than that of the slave.
400
@param to Destination of the data
401
@param from Source of the data
402
@param param_data Precision (upper) and decimal (lower) values
404
@return New pointer into memory based on from + length of the data
409
const unsigned char *
410
Field_decimal::unpack(unsigned char* to,
411
const unsigned char *from,
407
Field_new_decimal::unpack(uchar* to,
415
412
if (param_data == 0)
416
413
return Field::unpack(to, from, param_data, low_byte_first);
418
uint32_t from_precision= (param_data & 0xff00) >> 8U;
419
uint32_t from_decimal= param_data & 0x00ff;
420
uint32_t length=pack_length();
421
uint32_t from_pack_len= my_decimal_get_binary_size(from_precision, from_decimal);
422
uint32_t len= (param_data && (from_pack_len < length)) ?
423
from_pack_len : length;
415
uint from_precision= (param_data & 0xff00) >> 8U;
416
uint from_decimal= param_data & 0x00ff;
417
uint length=pack_length();
418
uint from_pack_len= my_decimal_get_binary_size(from_precision, from_decimal);
419
uint len= (param_data && (from_pack_len < length)) ?
420
from_pack_len : length;
424
421
if ((from_pack_len && (from_pack_len < length)) ||
425
422
(from_precision < precision) ||
426
423
(from_decimal < decimals()))