Commit 583a7452 authored by Marko Mäkelä's avatar Marko Mäkelä

Merge 10.6 into 10.11

parents 45fadb64 9c5600ad
...@@ -1646,6 +1646,17 @@ SELECT JSON_OBJECTAGG('\\', 1); ...@@ -1646,6 +1646,17 @@ SELECT JSON_OBJECTAGG('\\', 1);
JSON_OBJECTAGG('\\', 1) JSON_OBJECTAGG('\\', 1)
{"\\":1} {"\\":1}
# #
# MDEV-24784 JSON_ARRAYAGG charset issue
#
set names utf8;
select json_arrayagg('ä'), json_objectagg(1, 'ä');
json_arrayagg('ä') json_objectagg(1, 'ä')
["ä"] {"1":"ä"}
set names latin1;
select json_arrayagg('ä'), json_objectagg(1, 'ä');
json_arrayagg('ä') json_objectagg(1, 'ä')
["ä"] {"1":"ä"}
#
# End of 10.5 tests # End of 10.5 tests
# #
# #
......
...@@ -1111,6 +1111,16 @@ SELECT JSON_OBJECTAGG('"', 1); ...@@ -1111,6 +1111,16 @@ SELECT JSON_OBJECTAGG('"', 1);
SELECT JSON_OBJECTAGG('\"', 1); SELECT JSON_OBJECTAGG('\"', 1);
SELECT JSON_OBJECTAGG('\\', 1); SELECT JSON_OBJECTAGG('\\', 1);
--echo #
--echo # MDEV-24784 JSON_ARRAYAGG charset issue
--echo #
--disable_service_connection
set names utf8;
select json_arrayagg('ä'), json_objectagg(1, 'ä');
set names latin1;
select json_arrayagg('ä'), json_objectagg(1, 'ä');
--enable_service_connection
--echo # --echo #
--echo # End of 10.5 tests --echo # End of 10.5 tests
--echo # --echo #
......
...@@ -3,7 +3,7 @@ FLUSH TABLES; ...@@ -3,7 +3,7 @@ FLUSH TABLES;
# MDEV-11369: Instant ADD COLUMN for InnoDB # MDEV-11369: Instant ADD COLUMN for InnoDB
# #
CREATE TABLE t1(id INT PRIMARY KEY, c2 INT UNIQUE) CREATE TABLE t1(id INT PRIMARY KEY, c2 INT UNIQUE)
ENGINE=InnoDB ROW_FORMAT=REDUNDANT; ENGINE=InnoDB STATS_PERSISTENT=0 ROW_FORMAT=REDUNDANT;
CREATE TABLE t2 LIKE t1; CREATE TABLE t2 LIKE t1;
INSERT INTO t1 VALUES(0,2); INSERT INTO t1 VALUES(0,2);
INSERT INTO t2 VALUES(2,1); INSERT INTO t2 VALUES(2,1);
...@@ -27,7 +27,6 @@ SELECT * FROM t2; ...@@ -27,7 +27,6 @@ SELECT * FROM t2;
id c2 c3 id c2 c3
2 1 De finibus bonorum 2 1 De finibus bonorum
3 4 accusantium doloremque laudantium 3 4 accusantium doloremque laudantium
InnoDB 0 transactions not purged
BEGIN; BEGIN;
DELETE FROM t1; DELETE FROM t1;
ROLLBACK; ROLLBACK;
...@@ -161,7 +160,7 @@ t1 CREATE TABLE `t1` ( ...@@ -161,7 +160,7 @@ t1 CREATE TABLE `t1` (
`c2` int(11) DEFAULT NULL, `c2` int(11) DEFAULT NULL,
PRIMARY KEY (`id`), PRIMARY KEY (`id`),
UNIQUE KEY `c2` (`c2`) UNIQUE KEY `c2` (`c2`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1 COLLATE=latin1_swedish_ci ROW_FORMAT=REDUNDANT ) ENGINE=InnoDB DEFAULT CHARSET=latin1 COLLATE=latin1_swedish_ci STATS_PERSISTENT=0 ROW_FORMAT=REDUNDANT
SHOW CREATE TABLE t2; SHOW CREATE TABLE t2;
Table Create Table Table Create Table
t2 CREATE TABLE `t2` ( t2 CREATE TABLE `t2` (
...@@ -169,7 +168,7 @@ t2 CREATE TABLE `t2` ( ...@@ -169,7 +168,7 @@ t2 CREATE TABLE `t2` (
`c2` int(11) DEFAULT NULL, `c2` int(11) DEFAULT NULL,
PRIMARY KEY (`id`), PRIMARY KEY (`id`),
UNIQUE KEY `c2` (`c2`) UNIQUE KEY `c2` (`c2`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1 COLLATE=latin1_swedish_ci ROW_FORMAT=REDUNDANT ) ENGINE=InnoDB DEFAULT CHARSET=latin1 COLLATE=latin1_swedish_ci STATS_PERSISTENT=0 ROW_FORMAT=REDUNDANT
SHOW CREATE TABLE t3; SHOW CREATE TABLE t3;
Table Create Table Table Create Table
t3 CREATE TABLE `t3` ( t3 CREATE TABLE `t3` (
......
...@@ -14,7 +14,7 @@ let MYSQLD_DATADIR=`select @@datadir`; ...@@ -14,7 +14,7 @@ let MYSQLD_DATADIR=`select @@datadir`;
--echo # --echo #
CREATE TABLE t1(id INT PRIMARY KEY, c2 INT UNIQUE) CREATE TABLE t1(id INT PRIMARY KEY, c2 INT UNIQUE)
ENGINE=InnoDB ROW_FORMAT=REDUNDANT; ENGINE=InnoDB STATS_PERSISTENT=0 ROW_FORMAT=REDUNDANT;
CREATE TABLE t2 LIKE t1; CREATE TABLE t2 LIKE t1;
INSERT INTO t1 VALUES(0,2); INSERT INTO t1 VALUES(0,2);
INSERT INTO t2 VALUES(2,1); INSERT INTO t2 VALUES(2,1);
...@@ -38,7 +38,6 @@ disconnect ddl; ...@@ -38,7 +38,6 @@ disconnect ddl;
SELECT * FROM t1; SELECT * FROM t1;
SELECT * FROM t2; SELECT * FROM t2;
--source include/wait_all_purged.inc
BEGIN; BEGIN;
DELETE FROM t1; DELETE FROM t1;
ROLLBACK; ROLLBACK;
......
...@@ -4741,30 +4741,6 @@ bool Field_longlong::is_max() ...@@ -4741,30 +4741,6 @@ bool Field_longlong::is_max()
single precision float single precision float
****************************************************************************/ ****************************************************************************/
Field_float::Field_float(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg,
uchar null_bit_arg,
enum utype unireg_check_arg,
const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg,
bool zero_arg, bool unsigned_arg)
:Field_real(ptr_arg, len_arg, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg,
(dec_arg >= FLOATING_POINT_DECIMALS ? NOT_FIXED_DEC : dec_arg),
zero_arg, unsigned_arg)
{
}
Field_float::Field_float(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "": 0, (uint) 0,
NONE, field_name_arg,
(dec_arg >= FLOATING_POINT_DECIMALS ? NOT_FIXED_DEC : dec_arg),
0, 0)
{
}
int Field_float::store(const char *from,size_t len,CHARSET_INFO *cs) int Field_float::store(const char *from,size_t len,CHARSET_INFO *cs)
{ {
int error; int error;
...@@ -4913,40 +4889,6 @@ Binlog_type_info Field_float::binlog_type_info() const ...@@ -4913,40 +4889,6 @@ Binlog_type_info Field_float::binlog_type_info() const
double precision floating point numbers double precision floating point numbers
****************************************************************************/ ****************************************************************************/
Field_double::Field_double(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg,
uchar null_bit_arg,
enum utype unireg_check_arg,
const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg,
bool zero_arg, bool unsigned_arg)
:Field_real(ptr_arg, len_arg, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg,
(dec_arg >= FLOATING_POINT_DECIMALS ? NOT_FIXED_DEC : dec_arg),
zero_arg, unsigned_arg)
{
}
Field_double::Field_double(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "" : 0, (uint) 0,
NONE, field_name_arg,
(dec_arg >= FLOATING_POINT_DECIMALS ? NOT_FIXED_DEC : dec_arg),
0, 0)
{
}
Field_double::Field_double(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg, bool not_fixed_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "" : 0, (uint) 0,
NONE, field_name_arg,
(dec_arg >= FLOATING_POINT_DECIMALS ? NOT_FIXED_DEC : dec_arg),
0, 0)
{
not_fixed= not_fixed_arg;
}
int Field_double::store(const char *from,size_t len,CHARSET_INFO *cs) int Field_double::store(const char *from,size_t len,CHARSET_INFO *cs)
{ {
int error; int error;
......
...@@ -2137,7 +2137,7 @@ class Field_str :public Field { ...@@ -2137,7 +2137,7 @@ class Field_str :public Field {
const LEX_CSTRING *field_name_arg, const LEX_CSTRING *field_name_arg,
const DTCollation &collation); const DTCollation &collation);
decimal_digits_t decimals() const override decimal_digits_t decimals() const override
{ return is_created_from_null_item ? 0 : NOT_FIXED_DEC; } { return is_created_from_null_item ? 0 : DECIMAL_NOT_SPECIFIED; }
int save_in_field(Field *to) override { return save_in_field_str(to); } int save_in_field(Field *to) override { return save_in_field_str(to); }
bool memcpy_field_possible(const Field *from) const override bool memcpy_field_possible(const Field *from) const override
{ {
...@@ -2307,7 +2307,7 @@ class Field_real :public Field_num { ...@@ -2307,7 +2307,7 @@ class Field_real :public Field_num {
Information_schema_numeric_attributes Information_schema_numeric_attributes
information_schema_numeric_attributes() const override information_schema_numeric_attributes() const override
{ {
return dec == NOT_FIXED_DEC ? return dec == DECIMAL_NOT_SPECIFIED ?
Information_schema_numeric_attributes(field_length) : Information_schema_numeric_attributes(field_length) :
Information_schema_numeric_attributes(field_length, dec); Information_schema_numeric_attributes(field_length, dec);
} }
...@@ -2879,15 +2879,24 @@ class Field_vers_trx_id :public Field_longlong { ...@@ -2879,15 +2879,24 @@ class Field_vers_trx_id :public Field_longlong {
integers. But in all other cases we treat it as TIME_RESULT! */ integers. But in all other cases we treat it as TIME_RESULT! */
}; };
static inline decimal_digits_t fix_dec_arg(decimal_digits_t dec_arg)
{ return dec_arg >= FLOATING_POINT_DECIMALS ? DECIMAL_NOT_SPECIFIED : dec_arg; }
class Field_float final :public Field_real { class Field_float final :public Field_real {
public: public:
Field_float(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg, Field_float(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg,
uchar null_bit_arg, uchar null_bit_arg,
enum utype unireg_check_arg, const LEX_CSTRING *field_name_arg, enum utype unireg_check_arg, const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg,bool zero_arg,bool unsigned_arg); decimal_digits_t dec_arg,bool zero_arg,bool unsigned_arg)
:Field_real(ptr_arg, len_arg, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg,
fix_dec_arg(dec_arg), zero_arg, unsigned_arg)
{ }
Field_float(uint32 len_arg, bool maybe_null_arg, Field_float(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg, decimal_digits_t dec_arg); const LEX_CSTRING *field_name_arg, decimal_digits_t dec_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "": 0, (uint) 0,
NONE, field_name_arg, fix_dec_arg(dec_arg), 0, 0)
{ }
const Type_handler *type_handler() const override const Type_handler *type_handler() const override
{ return &type_handler_float; } { return &type_handler_float; }
enum ha_base_keytype key_type() const override { return HA_KEYTYPE_FLOAT; } enum ha_base_keytype key_type() const override { return HA_KEYTYPE_FLOAT; }
...@@ -2920,12 +2929,24 @@ class Field_double :public Field_real { ...@@ -2920,12 +2929,24 @@ class Field_double :public Field_real {
Field_double(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg, Field_double(uchar *ptr_arg, uint32 len_arg, uchar *null_ptr_arg,
uchar null_bit_arg, uchar null_bit_arg,
enum utype unireg_check_arg, const LEX_CSTRING *field_name_arg, enum utype unireg_check_arg, const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg,bool zero_arg,bool unsigned_arg); decimal_digits_t dec_arg,bool zero_arg,bool unsigned_arg)
:Field_real(ptr_arg, len_arg, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg,
fix_dec_arg(dec_arg), zero_arg, unsigned_arg)
{ }
Field_double(uint32 len_arg, bool maybe_null_arg, Field_double(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg, decimal_digits_t dec_arg); const LEX_CSTRING *field_name_arg, decimal_digits_t dec_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "" : 0, (uint) 0,
NONE, field_name_arg, fix_dec_arg(dec_arg), 0, 0)
{ }
Field_double(uint32 len_arg, bool maybe_null_arg, Field_double(uint32 len_arg, bool maybe_null_arg,
const LEX_CSTRING *field_name_arg, const LEX_CSTRING *field_name_arg,
decimal_digits_t dec_arg, bool not_fixed_arg); decimal_digits_t dec_arg, bool not_fixed_arg)
:Field_real((uchar*) 0, len_arg, maybe_null_arg ? (uchar*) "" : 0, (uint) 0,
NONE, field_name_arg, fix_dec_arg(dec_arg), 0, 0)
{
not_fixed= not_fixed_arg;
}
void init_for_tmp_table(Field *org_field, TABLE *new_table) override void init_for_tmp_table(Field *org_field, TABLE *new_table) override
{ {
Field::init_for_tmp_table(org_field, new_table); Field::init_for_tmp_table(org_field, new_table);
......
...@@ -4088,6 +4088,13 @@ int Arg_comparator::compare_e_json_str_basic(Item *j, Item *s) ...@@ -4088,6 +4088,13 @@ int Arg_comparator::compare_e_json_str_basic(Item *j, Item *s)
return MY_TEST(sortcmp(res1, res2, compare_collation()) == 0); return MY_TEST(sortcmp(res1, res2, compare_collation()) == 0);
} }
bool Item_func_json_arrayagg::fix_fields(THD *thd, Item **ref)
{
bool res= Item_func_group_concat::fix_fields(thd, ref);
m_tmp_json.set_charset(collation.collation);
return res;
}
String *Item_func_json_arrayagg::get_str_from_item(Item *i, String *tmp) String *Item_func_json_arrayagg::get_str_from_item(Item *i, String *tmp)
{ {
......
...@@ -717,6 +717,7 @@ class Item_func_json_arrayagg : public Item_func_group_concat ...@@ -717,6 +717,7 @@ class Item_func_json_arrayagg : public Item_func_group_concat
static LEX_CSTRING name= {STRING_WITH_LEN("json_arrayagg(") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_arrayagg(") };
return name; return name;
} }
bool fix_fields(THD *thd, Item **ref) override;
enum Sumfunctype sum_func() const override { return JSON_ARRAYAGG_FUNC; } enum Sumfunctype sum_func() const override { return JSON_ARRAYAGG_FUNC; }
String* val_str(String *str) override; String* val_str(String *str) override;
......
...@@ -350,14 +350,21 @@ trx_purge_add_undo_to_history(const trx_t* trx, trx_undo_t*& undo, mtr_t* mtr) ...@@ -350,14 +350,21 @@ trx_purge_add_undo_to_history(const trx_t* trx, trx_undo_t*& undo, mtr_t* mtr)
} }
/** Free an undo log segment. /** Free an undo log segment.
@param block rollback segment header page @param rseg_hdr rollback segment header page
@param block undo segment header page
@param mtr mini-transaction */ @param mtr mini-transaction */
static void trx_purge_free_segment(buf_block_t *block, mtr_t &mtr) static void trx_purge_free_segment(buf_block_t *rseg_hdr, buf_block_t *block,
mtr_t &mtr)
{ {
ut_ad(mtr.memo_contains_flagged(rseg_hdr, MTR_MEMO_PAGE_X_FIX));
ut_ad(mtr.memo_contains_flagged(block, MTR_MEMO_PAGE_X_FIX));
while (!fseg_free_step_not_header(TRX_UNDO_SEG_HDR + TRX_UNDO_FSEG_HEADER + while (!fseg_free_step_not_header(TRX_UNDO_SEG_HDR + TRX_UNDO_FSEG_HEADER +
block->page.frame, &mtr)) block->page.frame, &mtr))
{ {
rseg_hdr->fix();
block->fix(); block->fix();
ut_d(const page_id_t rseg_hdr_id{rseg_hdr->page.id()});
ut_d(const page_id_t id{block->page.id()}); ut_d(const page_id_t id{block->page.id()});
mtr.commit(); mtr.commit();
/* NOTE: If the server is killed after the log that was produced /* NOTE: If the server is killed after the log that was produced
...@@ -368,8 +375,11 @@ static void trx_purge_free_segment(buf_block_t *block, mtr_t &mtr) ...@@ -368,8 +375,11 @@ static void trx_purge_free_segment(buf_block_t *block, mtr_t &mtr)
This does not matter when using multiple innodb_undo_tablespaces; This does not matter when using multiple innodb_undo_tablespaces;
innodb_undo_log_truncate=ON will be able to reclaim the space. */ innodb_undo_log_truncate=ON will be able to reclaim the space. */
mtr.start(); mtr.start();
rseg_hdr->page.lock.x_lock();
ut_ad(rseg_hdr->page.id() == rseg_hdr_id);
block->page.lock.x_lock(); block->page.lock.x_lock();
ut_ad(block->page.id() == id); ut_ad(block->page.id() == id);
mtr.memo_push(rseg_hdr, MTR_MEMO_PAGE_X_MODIFY);
mtr.memo_push(block, MTR_MEMO_PAGE_X_MODIFY); mtr.memo_push(block, MTR_MEMO_PAGE_X_MODIFY);
} }
...@@ -459,7 +469,7 @@ trx_purge_truncate_rseg_history(trx_rseg_t &rseg, ...@@ -459,7 +469,7 @@ trx_purge_truncate_rseg_history(trx_rseg_t &rseg,
free_segment: free_segment:
ut_ad(rseg.curr_size >= seg_size); ut_ad(rseg.curr_size >= seg_size);
rseg.curr_size-= seg_size; rseg.curr_size-= seg_size;
trx_purge_free_segment(b, mtr); trx_purge_free_segment(rseg_hdr, b, mtr);
break; break;
case TRX_UNDO_CACHED: case TRX_UNDO_CACHED:
/* rseg.undo_cached must point to this page */ /* rseg.undo_cached must point to this page */
......
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