22
#include <drizzled/server_includes.h>
23
23
#include <drizzled/field/blob.h>
24
24
#include <drizzled/table.h>
25
25
#include <drizzled/session.h>
26
#include "plugin/myisam/myisam.h"
31
29
using namespace std;
36
static uint32_t blob_pack_length_to_max_length(uint32_t arg)
38
return max(UINT32_MAX,
39
(uint32_t)((INT64_C(1) << min(arg, 4U) * 8) - INT64_C(1)));
32
blob_pack_length_to_max_length(uint32_t arg)
34
return (INT64_C(1) << cmin(arg, 4U) * 8) - INT64_C(1);
43
38
/****************************************************************************
45
40
** A blob is saved as a length and a pointer. The length is stored in the
46
** packlength slot and is sizeof(uint32_t) (4 bytes)
41
** packlength slot and may be from 1-4.
47
42
****************************************************************************/
49
Field_blob::Field_blob(unsigned char *ptr_arg,
50
unsigned char *null_ptr_arg,
51
unsigned char null_bit_arg,
52
const char *field_name_arg,
54
const CHARSET_INFO * const cs)
56
blob_pack_length_to_max_length(sizeof(uint32_t)),
44
Field_blob::Field_blob(unsigned char *ptr_arg, unsigned char *null_ptr_arg, unsigned char null_bit_arg,
45
enum utype unireg_check_arg, const char *field_name_arg,
46
TABLE_SHARE *share, uint32_t blob_pack_length,
47
const CHARSET_INFO * const cs)
48
:Field_longstr(ptr_arg, blob_pack_length_to_max_length(blob_pack_length),
49
null_ptr_arg, null_bit_arg, unireg_check_arg, field_name_arg,
51
packlength(blob_pack_length)
63
54
share->blob_fields++;
64
/* TODO: why do not fill table->getShare()->blob_field array here? */
55
/* TODO: why do not fill table->s->blob_field array here? */
67
59
void Field_blob::store_length(unsigned char *i_ptr,
60
uint32_t i_packlength,
62
bool low_byte_first __attribute__((unused)))
71
#ifndef WORDS_BIGENDIAN
75
#ifdef WORDS_BIGENDIAN
78
int4store(i_ptr,i_number);
64
switch (i_packlength) {
66
i_ptr[0]= (unsigned char) i_number;
69
#ifdef WORDS_BIGENDIAN
72
int2store(i_ptr,(unsigned short) i_number);
76
shortstore(i_ptr,(unsigned short) i_number);
79
int3store(i_ptr,i_number);
82
#ifdef WORDS_BIGENDIAN
85
int4store(i_ptr,i_number);
89
longstore(i_ptr,i_number);
82
longstore(i_ptr,i_number);
86
void Field_blob::store_length(unsigned char *i_ptr, uint32_t i_number)
94
void Field_blob::store_length(unsigned char *i_ptr, uint32_t i_packlength,
88
store_length(i_ptr, i_number, getTable()->getShare()->db_low_byte_first);
97
store_length(i_ptr, i_packlength, i_number, table->s->db_low_byte_first);
92
101
uint32_t Field_blob::get_length(const unsigned char *pos,
102
uint32_t packlength_arg,
103
bool low_byte_first __attribute__((unused)))
95
#ifndef WORDS_BIGENDIAN
99
#ifdef WORDS_BIGENDIAN
105
return (uint32_t) tmp;
105
switch (packlength_arg) {
107
return (uint32_t) pos[0];
111
#ifdef WORDS_BIGENDIAN
117
return (uint32_t) tmp;
120
return (uint32_t) uint3korr(pos);
124
#ifdef WORDS_BIGENDIAN
130
return (uint32_t) tmp;
133
return 0; // Impossible
109
137
uint32_t Field_blob::get_packed_size(const unsigned char *ptr_arg,
110
138
bool low_byte_first)
112
return sizeof(uint32_t) + get_length(ptr_arg, low_byte_first);
140
return packlength + get_length(ptr_arg, packlength, low_byte_first);
116
144
uint32_t Field_blob::get_length(uint32_t row_offset)
118
return get_length(ptr+row_offset,
119
getTable()->getShare()->db_low_byte_first);
146
return get_length(ptr+row_offset, this->packlength,
147
table->s->db_low_byte_first);
123
151
uint32_t Field_blob::get_length(const unsigned char *ptr_arg)
125
return get_length(ptr_arg, getTable()->getShare()->db_low_byte_first);
153
return get_length(ptr_arg, this->packlength, table->s->db_low_byte_first);
130
158
Put a blob length field into a record buffer.
132
Blob length is always stored in sizeof(uint32_t) (4 bytes)
160
Depending on the maximum length of a blob, its length field is
161
put into 1 to 4 bytes. This is a property of the blob object,
162
described by 'packlength'.
134
164
@param pos Pointer into the record buffer.
135
165
@param length The length value to put.
209
261
int Field_blob::store(double nr)
211
263
const CHARSET_INFO * const cs=charset();
212
ASSERT_COLUMN_MARKED_FOR_WRITE;
213
264
value.set_real(nr, NOT_FIXED_DEC, cs);
214
return Field_blob::store(value.ptr(),(uint32_t) value.length(), cs);
265
return Field_blob::store(value.ptr(),(uint) value.length(), cs);
218
269
int Field_blob::store(int64_t nr, bool unsigned_val)
220
271
const CHARSET_INFO * const cs=charset();
221
ASSERT_COLUMN_MARKED_FOR_WRITE;
222
272
value.set_int(nr, unsigned_val, cs);
223
return Field_blob::store(value.ptr(), (uint32_t) value.length(), cs);
273
return Field_blob::store(value.ptr(), (uint) value.length(), cs);
250
ASSERT_COLUMN_MARKED_FOR_READ;
252
memcpy(&blob,ptr+sizeof(uint32_t),sizeof(char*));
297
memcpy(&blob,ptr+packlength,sizeof(char*));
255
300
uint32_t length=get_length(ptr);
256
301
return my_strntoll(charset(),blob,length,10,NULL,¬_used);
259
String *Field_blob::val_str(String *,
304
String *Field_blob::val_str(String *val_buffer __attribute__((unused)),
264
ASSERT_COLUMN_MARKED_FOR_READ;
266
memcpy(&blob,ptr+sizeof(uint32_t),sizeof(char*));
308
memcpy(&blob,ptr+packlength,sizeof(char*));
268
310
val_ptr->set("",0,charset()); // A bit safer than ->length(0)
321
358
int Field_blob::cmp_binary(const unsigned char *a_ptr, const unsigned char *b_ptr,
326
363
uint32_t a_length,b_length;
327
memcpy(&a,a_ptr+sizeof(uint32_t),sizeof(char*));
328
memcpy(&b,b_ptr+sizeof(uint32_t),sizeof(char*));
330
a_length= get_length(a_ptr);
364
memcpy(&a,a_ptr+packlength,sizeof(char*));
365
memcpy(&b,b_ptr+packlength,sizeof(char*));
366
a_length=get_length(a_ptr);
332
367
if (a_length > max_length)
333
a_length= max_length;
335
b_length= get_length(b_ptr);
369
b_length=get_length(b_ptr);
337
370
if (b_length > max_length)
338
b_length= max_length;
340
diff= memcmp(a,b,min(a_length,b_length));
342
return diff ? diff : (unsigned int) (a_length - b_length);
372
diff=memcmp(a,b,cmin(a_length,b_length));
373
return diff ? diff : (int) (a_length - b_length);
345
377
/* The following is used only when comparing a key */
346
uint32_t Field_blob::get_key_image(unsigned char *buff, uint32_t length)
379
uint32_t Field_blob::get_key_image(unsigned char *buff,
348
383
uint32_t blob_length= get_length(ptr);
349
384
unsigned char *blob;
397
434
return HA_KEY_BLOB_LENGTH+length;
400
438
void Field_blob::set_key_image(const unsigned char *buff,uint32_t length)
402
440
length= uint2korr(buff);
403
(void) Field_blob::store((const char*) buff+HA_KEY_BLOB_LENGTH, length, field_charset);
441
(void) Field_blob::store((const char*) buff+HA_KEY_BLOB_LENGTH, length,
406
446
int Field_blob::key_cmp(const unsigned char *key_ptr, uint32_t max_key_length)
408
448
unsigned char *blob1;
409
449
uint32_t blob_length=get_length(ptr);
410
memcpy(&blob1,ptr+sizeof(uint32_t),sizeof(char*));
450
memcpy(&blob1,ptr+packlength,sizeof(char*));
411
451
const CHARSET_INFO * const cs= charset();
412
452
uint32_t local_char_length= max_key_length / cs->mbmaxlen;
413
453
local_char_length= my_charpos(cs, blob1, blob1+blob_length,
424
464
b+HA_KEY_BLOB_LENGTH, uint2korr(b));
469
Save the field metadata for blob fields.
471
Saves the pack length in the first byte of the field metadata array
472
at index of *metadata_ptr.
474
@param metadata_ptr First byte of field metadata
476
@returns number of bytes written to metadata_ptr
478
int Field_blob::do_save_field_metadata(unsigned char *metadata_ptr)
480
*metadata_ptr= pack_length_no_ptr();
427
485
uint32_t Field_blob::sort_length() const
429
return (uint32_t) (getTable()->getSession()->variables.max_sort_length +
430
(field_charset == &my_charset_bin ? 0 : sizeof(uint32_t)));
487
return (uint32_t) (current_session->variables.max_sort_length +
488
(field_charset == &my_charset_bin ? 0 : packlength));
433
492
void Field_blob::sort_string(unsigned char *to,uint32_t length)
435
494
unsigned char *blob;
447
506
Store length of blob last in blob to shorter blobs before longer blobs
449
length-= sizeof(uint32_t); // size of stored blob length
452
mi_int4store(pos, blob_length);
511
switch (packlength) {
513
*pos= (char) blob_length;
516
mi_int2store(pos, blob_length);
519
mi_int3store(pos, blob_length);
522
mi_int4store(pos, blob_length);
454
memcpy(&blob,ptr+sizeof(uint32_t),sizeof(char*));
526
memcpy(&blob,ptr+packlength,sizeof(char*));
456
528
blob_length=my_strnxfrm(field_charset,
457
529
to, length, blob, blob_length);
475
549
unsigned char *Field_blob::pack(unsigned char *to, const unsigned char *from,
476
uint32_t max_length, bool low_byte_first)
550
uint32_t max_length, bool low_byte_first)
478
552
unsigned char *save= ptr;
479
553
ptr= (unsigned char*) from;
480
uint32_t length= get_length(); // Length of from string
554
uint32_t length=get_length(); // Length of from string
483
557
Store max length, which will occupy packlength bytes. If the max
484
558
length given is smaller than the actual length of the blob, we
485
559
just store the initial bytes of the blob.
487
store_length(to, min(length, max_length), low_byte_first);
561
store_length(to, packlength, cmin(length, max_length), low_byte_first);
490
564
Store the actual blob data, which will occupy 'length' bytes.
517
591
@return New pointer into memory based on from + length of the data
519
const unsigned char *Field_blob::unpack(unsigned char *,
520
const unsigned char *from,
593
const unsigned char *Field_blob::unpack(unsigned char *to __attribute__((unused)),
594
const unsigned char *from,
524
uint32_t const length= get_length(from, low_byte_first);
525
getTable()->setWriteSet(position());
526
store(reinterpret_cast<const char*>(from) + sizeof(uint32_t),
598
uint32_t const master_packlength=
599
param_data > 0 ? param_data & 0xFF : packlength;
600
uint32_t const length= get_length(from, master_packlength, low_byte_first);
601
bitmap_set_bit(table->write_set, field_index);
602
store(reinterpret_cast<const char*>(from) + master_packlength,
527
603
length, field_charset);
528
return(from + sizeof(uint32_t) + length);
604
return(from + master_packlength + length);
607
/* Keys for blobs are like keys on varchars */
609
int Field_blob::pack_cmp(const unsigned char *a, const unsigned char *b, uint32_t key_length_arg,
610
bool insert_or_update)
612
uint32_t a_length, b_length;
613
if (key_length_arg > 255)
615
a_length=uint2korr(a); a+=2;
616
b_length=uint2korr(b); b+=2;
620
a_length= (uint) *a++;
621
b_length= (uint) *b++;
623
return field_charset->coll->strnncollsp(field_charset,
630
int Field_blob::pack_cmp(const unsigned char *b, uint32_t key_length_arg,
631
bool insert_or_update)
634
uint32_t a_length, b_length;
635
memcpy(&a,ptr+packlength,sizeof(char*));
637
return key_length_arg > 0 ? -1 : 0;
639
a_length= get_length(ptr);
640
if (key_length_arg > 255)
642
b_length= uint2korr(b); b+=2;
645
b_length= (uint) *b++;
646
return field_charset->coll->strnncollsp(field_charset,
531
652
/** Create a packed key that will be used for storage from a MySQL row. */
534
655
Field_blob::pack_key(unsigned char *to, const unsigned char *from, uint32_t max_length,
656
bool low_byte_first __attribute__((unused)))
537
658
unsigned char *save= ptr;
538
659
ptr= (unsigned char*) from;
679
Unpack a blob key into a record buffer.
681
A blob key has a maximum size of 64K-1.
682
In its packed form, the length field is one or two bytes long,
683
depending on 'max_length'.
684
Depending on the maximum length of a blob, its length field is
685
put into 1 to 4 bytes. This is a property of the blob object,
686
described by 'packlength'.
687
Blobs are internally stored apart from the record buffer, which
688
contains a pointer to the blob buffer.
691
@param to Pointer into the record buffer.
692
@param from Pointer to the packed key.
693
@param max_length Key length limit from key description.
696
Pointer into 'from' past the last byte copied from packed key.
699
const unsigned char *
700
Field_blob::unpack_key(unsigned char *to, const unsigned char *from, uint32_t max_length,
701
bool low_byte_first __attribute__((unused)))
703
/* get length of the blob key */
704
uint32_t length= *from++;
705
if (max_length > 255)
706
length+= *from++ << 8;
708
/* put the length into the record buffer */
709
put_length(to, length);
711
/* put the address of the blob buffer or NULL */
713
memcpy(to + packlength, &from, sizeof(from));
715
memset(to + packlength, 0, sizeof(from));
717
/* point to first byte of next field in 'from' */
718
return from + length;
722
/** Create a packed key that will be used for storage from a MySQL key. */
725
Field_blob::pack_key_from_key_image(unsigned char *to, const unsigned char *from, uint32_t max_length,
726
bool low_byte_first __attribute__((unused)))
728
uint32_t length=uint2korr(from);
729
if (length > max_length)
731
*to++= (char) (length & 255);
732
if (max_length > 255)
733
*to++= (char) (length >> 8);
735
memcpy(to, from+HA_KEY_BLOB_LENGTH, length);
740
uint32_t Field_blob::packed_col_length(const unsigned char *data_ptr, uint32_t length)
743
return uint2korr(data_ptr)+2;
744
return (uint) *data_ptr + 1;
748
uint32_t Field_blob::max_packed_col_length(uint32_t max_length)
750
return (max_length > 255 ? 2 : 1)+max_length;
754
uint32_t Field_blob::is_equal(Create_field *new_field)
756
if (compare_str_field_flags(new_field, flags))
759
return ((new_field->sql_type == get_blob_type_from_length(max_data_length()))
760
&& new_field->charset == field_charset &&
761
((Field_blob *)new_field->field)->max_data_length() ==
558
767
maximum possible display length for blob.