Commit a14a34be authored by unknown's avatar unknown

InnoDB: On UPDATE, trim UCS2 columns correctly. (Bug #12178)


sql/ha_innodb.cc:
  innobase_convert_and_store_changed_col(), calc_row_difference():
   Replace parameter "is_unsigned" with "prtype".
  innobase_convert_and_store_changed_col():
   When trimming spaces, note that UCS2 spaces are 0x0020, not 0x20.
parent b96e4b68
...@@ -2614,7 +2614,7 @@ innobase_convert_and_store_changed_col( ...@@ -2614,7 +2614,7 @@ innobase_convert_and_store_changed_col(
mysql_byte* data, /* in: column data to store */ mysql_byte* data, /* in: column data to store */
ulint len, /* in: data len */ ulint len, /* in: data len */
ulint col_type,/* in: data type in InnoDB type numbers */ ulint col_type,/* in: data type in InnoDB type numbers */
ulint is_unsigned)/* in: != 0 if an unsigned integer type */ ulint prtype) /* InnoDB precise data type and flags */
{ {
uint i; uint i;
...@@ -2622,10 +2622,31 @@ innobase_convert_and_store_changed_col( ...@@ -2622,10 +2622,31 @@ innobase_convert_and_store_changed_col(
data = NULL; data = NULL;
} else if (col_type == DATA_VARCHAR || col_type == DATA_BINARY } else if (col_type == DATA_VARCHAR || col_type == DATA_BINARY
|| col_type == DATA_VARMYSQL) { || col_type == DATA_VARMYSQL) {
/* Remove trailing spaces */ /* Remove trailing spaces. */
while (len > 0 && data[len - 1] == ' ') {
/* Handle UCS2 strings differently. As no new
collations will be introduced in 4.1, we hardcode the
charset-collation codes here. In 5.0, the logic will
be based on mbminlen. */
ulint cset = dtype_get_charset_coll_noninline(prtype);
if (cset == 35/*ucs2_general_ci*/
|| cset == 90/*ucs2_bin*/
|| (cset >= 128/*ucs2_unicode_ci*/
&& cset <= 144/*ucs2_persian_ci*/)) {
/* space=0x0020 */
/* Trim "half-chars", just in case. */
len &= ~1;
while (len && data[len - 2] == 0x00
&& data[len - 1] == 0x20) {
len -= 2;
}
} else {
/* space=0x20 */
while (len && data[len - 1] == 0x20) {
len--; len--;
} }
}
} else if (col_type == DATA_INT) { } else if (col_type == DATA_INT) {
/* Store integer data in InnoDB in a big-endian /* Store integer data in InnoDB in a big-endian
format, sign bit negated, if signed */ format, sign bit negated, if signed */
...@@ -2634,7 +2655,7 @@ innobase_convert_and_store_changed_col( ...@@ -2634,7 +2655,7 @@ innobase_convert_and_store_changed_col(
buf[len - 1 - i] = data[i]; buf[len - 1 - i] = data[i];
} }
if (!is_unsigned) { if (!(prtype & DATA_UNSIGNED)) {
buf[0] = buf[0] ^ 128; buf[0] = buf[0] ^ 128;
} }
...@@ -2677,7 +2698,7 @@ calc_row_difference( ...@@ -2677,7 +2698,7 @@ calc_row_difference(
byte* buf; byte* buf;
upd_field_t* ufield; upd_field_t* ufield;
ulint col_type; ulint col_type;
ulint is_unsigned; ulint prtype;
ulint n_changed = 0; ulint n_changed = 0;
uint i; uint i;
...@@ -2702,8 +2723,7 @@ calc_row_difference( ...@@ -2702,8 +2723,7 @@ calc_row_difference(
n_len = field->pack_length(); n_len = field->pack_length();
col_type = prebuilt->table->cols[i].type.mtype; col_type = prebuilt->table->cols[i].type.mtype;
is_unsigned = prebuilt->table->cols[i].type.prtype & prtype = prebuilt->table->cols[i].type.prtype;
DATA_UNSIGNED;
switch (col_type) { switch (col_type) {
case DATA_BLOB: case DATA_BLOB:
...@@ -2743,7 +2763,7 @@ calc_row_difference( ...@@ -2743,7 +2763,7 @@ calc_row_difference(
innobase_convert_and_store_changed_col(ufield, innobase_convert_and_store_changed_col(ufield,
(mysql_byte*)buf, (mysql_byte*)buf,
(mysql_byte*)n_ptr, n_len, col_type, (mysql_byte*)n_ptr, n_len, col_type,
is_unsigned); prtype);
ufield->exp = NULL; ufield->exp = NULL;
ufield->field_no = prebuilt->table->cols[i].clust_pos; ufield->field_no = prebuilt->table->cols[i].clust_pos;
n_changed++; n_changed++;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment