Commit f071b762 authored by Yuchen Pei's avatar Yuchen Pei

Merge branch '10.5' into 10.6

parents 6264950c cf1c381b
...@@ -1070,7 +1070,7 @@ Exit_status process_event(PRINT_EVENT_INFO *print_event_info, Log_event *ev, ...@@ -1070,7 +1070,7 @@ Exit_status process_event(PRINT_EVENT_INFO *print_event_info, Log_event *ev,
case QUERY_COMPRESSED_EVENT: case QUERY_COMPRESSED_EVENT:
{ {
Query_log_event *qe= (Query_log_event*)ev; Query_log_event *qe= (Query_log_event*)ev;
if (!qe->is_trans_keyword()) if (!qe->is_trans_keyword(print_event_info->is_xa_trans()))
{ {
if (shall_skip_database(qe->db)) if (shall_skip_database(qe->db))
goto end; goto end;
......
...@@ -573,6 +573,22 @@ privilege\&. ...@@ -573,6 +573,22 @@ privilege\&.
.sp -1 .sp -1
.IP \(bu 2.3 .IP \(bu 2.3
.\} .\}
.\" mysqlbinlog: flashback option
.\" flashback option: mysqlbinlog
\fB\-\-flashback\fR,
\fB\-B\fR
.sp
Support flashback mode\&.
.RE
.sp
.RS 4
.ie n \{\
\h'-04'\(bu\h'+03'\c
.\}
.el \{\
.sp -1
.IP \(bu 2.3
.\}
.\" mysqlbinlog: force-if-open option .\" mysqlbinlog: force-if-open option
.\" force-if-open option: mysqlbinlog .\" force-if-open option: mysqlbinlog
\fB\-\-force\-if\-open\fR \fB\-\-force\-if\-open\fR
......
...@@ -894,22 +894,6 @@ instead\&. ...@@ -894,22 +894,6 @@ instead\&.
.sp -1 .sp -1
.IP \(bu 2.3 .IP \(bu 2.3
.\} .\}
.\" mysqldump: flashback option
.\" flashback option: mysqldump
\fB\-\-flashback\fR,
\fB\-B\fR
.sp
Support flashback mode\&.
.RE
.sp
.RS 4
.ie n \{\
\h'-04'\(bu\h'+03'\c
.\}
.el \{\
.sp -1
.IP \(bu 2.3
.\}
.\" mysqldump: flush-logs option .\" mysqldump: flush-logs option
.\" flush-logs option: mysqldump .\" flush-logs option: mysqldump
\fB\-\-flush\-logs\fR, \fB\-\-flush\-logs\fR,
......
...@@ -4,4 +4,4 @@ ...@@ -4,4 +4,4 @@
# - r_engine_stats depends on buffer pool state and whether old record versions # - r_engine_stats depends on buffer pool state and whether old record versions
# were purged. # were purged.
--replace_regex /("(r_[a-z_]*_time_ms|r_buffer_size)": )[^, \n]*/\1"REPLACED"/ /("r_engine_stats":) {[^}]*}/\1 REPLACED/ --replace_regex /("(r_[a-z_]*_time_ms|r_buffer_size|r_partial_match_buffer_size)": )[^, \n]*/\1"REPLACED"/ /("r_engine_stats":) {[^}]*}/\1 REPLACED/
if (`select version() like '%valgrind%' || version() like '%asan%'`)
{
skip Does not run with binaries built with valgrind or asan;
}
...@@ -692,6 +692,9 @@ ANALYZE ...@@ -692,6 +692,9 @@ ANALYZE
}, },
"subqueries": [ "subqueries": [
{ {
"materialization": {
"r_strategy": "index_lookup",
"r_loops": 2,
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"r_loops": 1, "r_loops": 1,
...@@ -710,6 +713,7 @@ ANALYZE ...@@ -710,6 +713,7 @@ ANALYZE
} }
} }
} }
}
] ]
} }
} }
......
...@@ -7671,6 +7671,7 @@ EXPLAIN ...@@ -7671,6 +7671,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"table": { "table": {
...@@ -7678,6 +7679,7 @@ EXPLAIN ...@@ -7678,6 +7679,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"read_sorted_file": { "read_sorted_file": {
"filesort": { "filesort": {
...@@ -7731,6 +7733,7 @@ EXPLAIN ...@@ -7731,6 +7733,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"table": { "table": {
...@@ -7738,6 +7741,7 @@ EXPLAIN ...@@ -7738,6 +7741,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived3>", "table_name": "<derived3>",
...@@ -7815,6 +7819,7 @@ EXPLAIN ...@@ -7815,6 +7819,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -7822,6 +7827,7 @@ EXPLAIN ...@@ -7822,6 +7827,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -7872,6 +7878,7 @@ EXPLAIN ...@@ -7872,6 +7878,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -7879,6 +7886,7 @@ EXPLAIN ...@@ -7879,6 +7886,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -7924,6 +7932,7 @@ EXPLAIN ...@@ -7924,6 +7932,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -7931,6 +7940,7 @@ EXPLAIN ...@@ -7931,6 +7940,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -7976,6 +7986,7 @@ EXPLAIN ...@@ -7976,6 +7986,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -7983,6 +7994,7 @@ EXPLAIN ...@@ -7983,6 +7994,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -8030,6 +8042,7 @@ EXPLAIN ...@@ -8030,6 +8042,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -8037,6 +8050,7 @@ EXPLAIN ...@@ -8037,6 +8050,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -8082,6 +8096,7 @@ EXPLAIN ...@@ -8082,6 +8096,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -8089,6 +8104,7 @@ EXPLAIN ...@@ -8089,6 +8104,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -8134,6 +8150,7 @@ EXPLAIN ...@@ -8134,6 +8150,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"table": { "table": {
...@@ -8141,6 +8158,7 @@ EXPLAIN ...@@ -8141,6 +8158,7 @@ EXPLAIN
} }
} }
} }
}
}, },
"table": { "table": {
"table_name": "<derived2>", "table_name": "<derived2>",
...@@ -8223,6 +8241,7 @@ EXPLAIN ...@@ -8223,6 +8241,7 @@ EXPLAIN
}, },
"subqueries": [ "subqueries": [
{ {
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"table": { "table": {
...@@ -8230,6 +8249,7 @@ EXPLAIN ...@@ -8230,6 +8249,7 @@ EXPLAIN
} }
} }
} }
}
] ]
} }
} }
...@@ -9294,6 +9314,7 @@ EXPLAIN ...@@ -9294,6 +9314,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"temporary_table": { "temporary_table": {
...@@ -9326,6 +9347,7 @@ EXPLAIN ...@@ -9326,6 +9347,7 @@ EXPLAIN
} }
} }
} }
}
} }
SELECT * FROM t1 SELECT * FROM t1
WHERE (t1.a,t1.b) IN WHERE (t1.a,t1.b) IN
...@@ -9396,6 +9418,7 @@ EXPLAIN ...@@ -9396,6 +9418,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"temporary_table": { "temporary_table": {
...@@ -9428,6 +9451,7 @@ EXPLAIN ...@@ -9428,6 +9451,7 @@ EXPLAIN
} }
} }
} }
}
} }
DROP TABLE t1,t2; DROP TABLE t1,t2;
# #
...@@ -10827,6 +10851,7 @@ EXPLAIN ...@@ -10827,6 +10851,7 @@ EXPLAIN
"attached_condition": "t4.c = `<subquery2>`.`sum(b)`", "attached_condition": "t4.c = `<subquery2>`.`sum(b)`",
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"having_condition": "`f1(a)` > 1 and `sum(b)` > 123", "having_condition": "`f1(a)` > 1 and `sum(b)` > 123",
...@@ -10843,6 +10868,7 @@ EXPLAIN ...@@ -10843,6 +10868,7 @@ EXPLAIN
} }
} }
} }
}
} }
drop view v2; drop view v2;
drop function f1; drop function f1;
......
...@@ -574,6 +574,7 @@ EXPLAIN ...@@ -574,6 +574,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"temporary_table": { "temporary_table": {
...@@ -588,6 +589,7 @@ EXPLAIN ...@@ -588,6 +589,7 @@ EXPLAIN
} }
} }
} }
}
} }
# #
# Semi-join Materialization # Semi-join Materialization
...@@ -906,6 +908,7 @@ EXPLAIN ...@@ -906,6 +908,7 @@ EXPLAIN
}, },
"subqueries": [ "subqueries": [
{ {
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"table": { "table": {
...@@ -916,6 +919,7 @@ EXPLAIN ...@@ -916,6 +919,7 @@ EXPLAIN
} }
} }
} }
}
] ]
} }
} }
......
...@@ -1134,6 +1134,7 @@ EXPLAIN ...@@ -1134,6 +1134,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"temporary_table": { "temporary_table": {
...@@ -1151,6 +1152,7 @@ EXPLAIN ...@@ -1151,6 +1152,7 @@ EXPLAIN
} }
} }
} }
}
} }
set statement optimizer_switch='condition_pushdown_from_having=off' for explain format=json SELECT * FROM t1 set statement optimizer_switch='condition_pushdown_from_having=off' for explain format=json SELECT * FROM t1
WHERE WHERE
...@@ -1183,6 +1185,7 @@ EXPLAIN ...@@ -1183,6 +1185,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"temporary_table": { "temporary_table": {
...@@ -1200,6 +1203,7 @@ EXPLAIN ...@@ -1200,6 +1203,7 @@ EXPLAIN
} }
} }
} }
}
} }
# conjunctive subformula : pushdown into materialized IN subquery # conjunctive subformula : pushdown into materialized IN subquery
# HAVING clause # HAVING clause
...@@ -1257,6 +1261,7 @@ EXPLAIN ...@@ -1257,6 +1261,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 2, "select_id": 2,
"having_condition": "`MAX(t2.y)` < 14", "having_condition": "`MAX(t2.y)` < 14",
...@@ -1275,6 +1280,7 @@ EXPLAIN ...@@ -1275,6 +1280,7 @@ EXPLAIN
} }
} }
} }
}
} }
set statement optimizer_switch='condition_pushdown_from_having=off' for explain format=json SELECT * FROM t1 set statement optimizer_switch='condition_pushdown_from_having=off' for explain format=json SELECT * FROM t1
WHERE WHERE
...@@ -1307,6 +1313,7 @@ EXPLAIN ...@@ -1307,6 +1313,7 @@ EXPLAIN
"filtered": 100, "filtered": 100,
"materialized": { "materialized": {
"unique": 1, "unique": 1,
"materialization": {
"query_block": { "query_block": {
"select_id": 3, "select_id": 3,
"having_condition": "`MAX(t2.y)` < 14", "having_condition": "`MAX(t2.y)` < 14",
...@@ -1325,6 +1332,7 @@ EXPLAIN ...@@ -1325,6 +1332,7 @@ EXPLAIN
} }
} }
} }
}
} }
# non-standard allowed queries # non-standard allowed queries
# conjunctive subformula # conjunctive subformula
...@@ -5088,4 +5096,91 @@ SELECT * FROM v1 ...@@ -5088,4 +5096,91 @@ SELECT * FROM v1
GROUP BY a HAVING a = (a IS NULL OR a IS NULL); GROUP BY a HAVING a = (a IS NULL OR a IS NULL);
a a
DROP VIEW v1; DROP VIEW v1;
#
# MDEV-32608: Expression with constant subquery causes a crash
# in pushdown from HAVING
#
CREATE TABLE t1 (a INT, b INT);
INSERT INTO t1 VALUES (2, 1), (3, 2);
EXPLAIN FORMAT=JSON SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a + b = 3;
EXPLAIN
{
"query_block": {
"select_id": 1,
"having_condition": "t1.a = (subquery#2)",
"filesort": {
"sort_key": "t1.b",
"temporary_table": {
"table": {
"table_name": "t1",
"access_type": "ALL",
"rows": 2,
"filtered": 100,
"attached_condition": "(subquery#2) + t1.b = 3"
},
"subqueries": [
{
"query_block": {
"select_id": 2,
"table": {
"table_name": "t1",
"access_type": "ALL",
"rows": 2,
"filtered": 100
}
}
}
]
}
}
}
}
SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a + b = 3;
a b
2 1
EXPLAIN FORMAT=JSON SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a > b;
EXPLAIN
{
"query_block": {
"select_id": 1,
"having_condition": "t1.a = (subquery#2)",
"filesort": {
"sort_key": "t1.b",
"temporary_table": {
"table": {
"table_name": "t1",
"access_type": "ALL",
"rows": 2,
"filtered": 100,
"attached_condition": "(subquery#2) > t1.b"
},
"subqueries": [
{
"query_block": {
"select_id": 2,
"table": {
"table_name": "t1",
"access_type": "ALL",
"rows": 2,
"filtered": 100
}
}
}
]
}
}
}
}
SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a > b;
a b
2 1
DROP TABLE t1;
End of 10.5 tests End of 10.5 tests
...@@ -1562,4 +1562,30 @@ SELECT * FROM v1 ...@@ -1562,4 +1562,30 @@ SELECT * FROM v1
DROP VIEW v1; DROP VIEW v1;
--echo #
--echo # MDEV-32608: Expression with constant subquery causes a crash
--echo # in pushdown from HAVING
--echo #
CREATE TABLE t1 (a INT, b INT);
INSERT INTO t1 VALUES (2, 1), (3, 2);
let $q=
SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a + b = 3;
eval EXPLAIN FORMAT=JSON $q;
eval $q;
let $q=
SELECT * FROM t1
GROUP BY b
HAVING (SELECT MAX(b) FROM t1) = a AND a > b;
eval EXPLAIN FORMAT=JSON $q;
eval $q;
DROP TABLE t1;
--echo End of 10.5 tests --echo End of 10.5 tests
--source include/not_msan.inc --source include/not_msan.inc
--source include/not_valgrind_build.inc --source include/not_valgrind.inc
--echo # MDEV-20699 do not cache SP in SHOW CREATE --echo # MDEV-20699 do not cache SP in SHOW CREATE
--echo # Warmup round, this might allocate some memory for session variable --echo # Warmup round, this might allocate some memory for session variable
......
This diff is collapsed.
set @save_optimizer_switch=@@optimizer_switch;
create table t1 (a int);
create table t2 (b int);
insert into t1 values (null), (1), (2), (3);
insert into t2 values (3), (4);
set @@optimizer_switch = "materialization=on,in_to_exists=off,semijoin=off";
explain format=json select * from t1 where a in (select b from t2);
--echo # "Complete match" execution strategy
--source include/analyze-format.inc
analyze format=json select * from t1 where a in (select b from t2);
--echo # "Partial match" is used due to NOT IN
--echo # Force rowid-merge partial partial matching
set @@optimizer_switch="partial_match_rowid_merge=on,partial_match_table_scan=off";
--source include/analyze-format.inc
analyze format=json select * from t1 where a not in (select b from t2);
--echo # Force table scan partial matching
set @@optimizer_switch="partial_match_rowid_merge=off,partial_match_table_scan=on";
--source include/analyze-format.inc
analyze format=json select * from t1 where a not in (select b from t2);
--echo # Subselect in GROUP BY
--source include/analyze-format.inc
analyze format=json select a from t1 group by a in (select b from t2);
set @@optimizer_switch="partial_match_rowid_merge=on,partial_match_table_scan=off";
--source include/analyze-format.inc
analyze format=json select a from t1 group by a not in (select b from t2);
set @@optimizer_switch="partial_match_rowid_merge=on,partial_match_table_scan=on";
--echo # Subselect in ORDER BY
--source include/analyze-format.inc
analyze format=json select a from t1 order by a in (select b from t2);
--echo # Subselect in HAVING
--source include/analyze-format.inc
analyze format=json select a from t1 having a not in (select b from t2);
--echo # Nested IN
--source include/analyze-format.inc
analyze format=json select a from t1 where a in (select a from t1 where a in (select b from t2));
create table t3 (c int);
insert into t3 (c) values (3), (null), (4);
--echo # Subquery in ON-clause of outer join
--source include/analyze-format.inc
analyze format=json select a from t1 left join t2 on a not in (select c from t3);
--source include/analyze-format.inc
analyze format=json
select (b, b + 1, b + 2) not in
(select count(distinct a), a + 1, a + 2 from t1 group by a + 1, a + 2)
from t2;
drop table t1, t2, t3;
--echo #
--echo # Tables with more than one column
--echo #
create table t1 (a1 char(1), a2 char(1));
insert into t1 values (null, 'b');
create table t2 (b1 char(1), b2 char(2));
insert into t2 values ('a','b'), ('c', 'd'), (null, 'e'), ('f', 'g');
set @@optimizer_switch="partial_match_rowid_merge=on,partial_match_table_scan=off";
explain format=json select * from t1 where (a1, a2) not in (select b1, b2 from t2);
--source include/analyze-format.inc
analyze format=json select * from t1 where (a1, a2) not in (select b1, b2 from t2);
set @@optimizer_switch="partial_match_rowid_merge=off,partial_match_table_scan=on";
--source include/analyze-format.inc
analyze format=json select * from t1 where (a1, a2) not in (select b1, b2 from t2);
--echo # Subquery in SELECT list
explain format=json select t1.*, (a1, a2) in (select * from t2) as in_res from t1;
--source include/analyze-format.inc
analyze format=json select t1.*, (a1, a2) in (select * from t2) as in_res from t1;
--source include/analyze-format.inc
analyze format=json select t1.*, (a1, a2) not in (select * from t2) as in_res from t1;
set @@optimizer_switch="partial_match_rowid_merge=on,partial_match_table_scan=off";
--source include/analyze-format.inc
analyze format=json select t1.*, (a1, a2) in (select * from t2) as in_res from t1;
--source include/analyze-format.inc
analyze format=json select t1.*, (a1, a2) not in (select * from t2) as in_res from t1;
drop table t1,t2;
set @@optimizer_switch=@save_optimizer_switch;
#
# MDEV-34542 Assertion `lock_trx_has_sys_table_locks(trx) == __null'
# failed in void row_mysql_unfreeze_data_dictionary(trx_t*)
#
#
CREATE TABLE t1 (c1 CHAR(1) ,c2 INT) ENGINE=INNODB
PARTITION BY LINEAR HASH ((c2)) PARTITIONS 512;
CREATE TABLE t2 (a INT) ENGINE=INNODB;
set @old_table_open_cache= @@table_open_cache;
XA START 'a';
INSERT INTO mysql.innodb_index_stats SELECT * FROM mysql.innodb_index_stats WHERE table_name='';
SET GLOBAL table_open_cache=10;
INSERT into t2 (a) VALUES (1);
SELECT * FROM t1;
c1 c2
XA END 'a';
XA PREPARE 'a';
SELECT sleep(3);
sleep(3)
0
XA ROLLBACK 'a';
DROP TABLE t1, t2;
SET GLOBAL table_open_cache=@old_table_open_cache;
--source include/have_innodb.inc
--source include/have_partition.inc
--echo #
--echo # MDEV-34542 Assertion `lock_trx_has_sys_table_locks(trx) == __null'
--echo # failed in void row_mysql_unfreeze_data_dictionary(trx_t*)
--echo #
--echo #
CREATE TABLE t1 (c1 CHAR(1) ,c2 INT) ENGINE=INNODB
PARTITION BY LINEAR HASH ((c2)) PARTITIONS 512;
CREATE TABLE t2 (a INT) ENGINE=INNODB;
set @old_table_open_cache= @@table_open_cache;
XA START 'a';
INSERT INTO mysql.innodb_index_stats SELECT * FROM mysql.innodb_index_stats WHERE table_name='';
SET GLOBAL table_open_cache=10;
INSERT into t2 (a) VALUES (1);
SELECT * FROM t1;
XA END 'a';
XA PREPARE 'a';
# Added sleep to make sure that InnoDB main thread is to remove
# the innodb_index_stats from table cache
SELECT sleep(3);
XA ROLLBACK 'a';
DROP TABLE t1, t2;
SET GLOBAL table_open_cache=@old_table_open_cache;
...@@ -1164,6 +1164,84 @@ include/sync_with_master_gtid.inc ...@@ -1164,6 +1164,84 @@ include/sync_with_master_gtid.inc
connection server_1; connection server_1;
set @@binlog_format = @sav_binlog_format; set @@binlog_format = @sav_binlog_format;
set @@global.binlog_format = @sav_binlog_format; set @@global.binlog_format = @sav_binlog_format;
#
# MDEV-33921.1: If a slave's replication of an XA transaction results in
# an empty transaction, e.g. due to replication filters, the slave
# should not binlog any part of the XA transaction.
connection server_1;
create database db1;
create database db2;
create table db1.t1 (a int) engine=innodb;
include/save_master_gtid.inc
connection server_3;
include/sync_with_master_gtid.inc
include/stop_slave.inc
connection server_2;
include/stop_slave.inc
SET @@GLOBAL.replicate_ignore_db= "";
SET @@GLOBAL.replicate_do_db= "db2";
include/start_slave.inc
connection server_1;
use db1;
XA START "x1";
insert into db1.t1 values (1);
XA END "x1";
XA PREPARE "x1";
XA COMMIT "x1";
include/save_master_gtid.inc
connection server_2;
include/sync_with_master_gtid.inc
connection server_2;
include/save_master_gtid.inc
connection server_3;
include/start_slave.inc
include/sync_with_master_gtid.inc
#
# 33921.2: If the slave shuts down after "preparing" a filtered-to-empty
# XA transaction (and not completing it), then when the respective
# XA completion (COMMIT in this test) command is replicated, the slave
# should not throw ER_XAER_NOTA. Note that internally, the error is
# thrown, but it is ignored because the target db is filtered.
connection server_3;
include/stop_slave.inc
connection server_1;
use db1;
XA START "x2";
insert into db1.t1 values (2);
XA END "x2";
XA PREPARE "x2";
include/save_master_gtid.inc
connection server_2;
include/sync_with_master_gtid.inc
# Connection named slave is needed for reconnection
connect slave,localhost,root,,;
connect slave1,localhost,root,,;
include/rpl_restart_server.inc [server_number=2]
connection server_2;
include/stop_slave.inc
SET @@GLOBAL.replicate_do_db= "db2";
include/start_slave.inc
connection server_1;
XA COMMIT "x2";
connection server_2;
include/sync_with_master_gtid.inc
include/save_master_gtid.inc
connection server_3;
include/start_slave.inc
include/sync_with_master_gtid.inc
#
# 33921.3: Ensure XA commands are not considered by mysqlbinlog's
# --database filter
connection server_1;
# MYSQL_BINLOG datadir/binlog_file --start-position=pre_xa_pos --database=db2 --result-file=assert_file
include/assert_grep.inc [Mysqlbinlog should output all XA commands from the filtered transaction]
connection server_2;
include/stop_slave.inc
SET @@GLOBAL.replicate_do_db="";
include/start_slave.inc
connection server_1;
drop database db1;
drop database db2;
connection server_1; connection server_1;
include/rpl_end.inc include/rpl_end.inc
# End of rpl_xa_empty_transaction.test # End of rpl_xa_empty_transaction.test
...@@ -55,9 +55,7 @@ SET sql_log_bin=1; ...@@ -55,9 +55,7 @@ SET sql_log_bin=1;
# Restart the slave mysqld server, and verify that the GTID position is # Restart the slave mysqld server, and verify that the GTID position is
# read correctly from the new mysql.gtid_slave_pos_innodb table. # read correctly from the new mysql.gtid_slave_pos_innodb table.
--write_file $MYSQLTEST_VARDIR/tmp/mysqld.2.expect --write_line wait $MYSQLTEST_VARDIR/tmp/mysqld.2.expect
wait
EOF
--shutdown_server --shutdown_server
--source include/wait_until_disconnected.inc --source include/wait_until_disconnected.inc
...@@ -94,9 +92,7 @@ DROP TABLE mysql.gtid_slave_pos; ...@@ -94,9 +92,7 @@ DROP TABLE mysql.gtid_slave_pos;
RENAME TABLE mysql.gtid_slave_pos_innodb TO mysql.gtid_slave_pos; RENAME TABLE mysql.gtid_slave_pos_innodb TO mysql.gtid_slave_pos;
SET sql_log_bin=1; SET sql_log_bin=1;
--write_file $MYSQLTEST_VARDIR/tmp/mysqld.2.expect --write_line wait $MYSQLTEST_VARDIR/tmp/mysqld.2.expect
wait
EOF
--shutdown_server --shutdown_server
--source include/wait_until_disconnected.inc --source include/wait_until_disconnected.inc
...@@ -132,9 +128,7 @@ SET sql_log_bin=0; ...@@ -132,9 +128,7 @@ SET sql_log_bin=0;
ALTER TABLE mysql.gtid_slave_pos ENGINE=Aria; ALTER TABLE mysql.gtid_slave_pos ENGINE=Aria;
SET sql_log_bin=1; SET sql_log_bin=1;
--write_file $MYSQLTEST_VARDIR/tmp/mysqld.2.expect --write_line wait $MYSQLTEST_VARDIR/tmp/mysqld.2.expect
wait
EOF
--shutdown_server --shutdown_server
--source include/wait_until_disconnected.inc --source include/wait_until_disconnected.inc
...@@ -176,9 +170,7 @@ INSERT INTO mysql.gtid_slave_pos SELECT * FROM mysql.gtid_slave_pos_InnoDB; ...@@ -176,9 +170,7 @@ INSERT INTO mysql.gtid_slave_pos SELECT * FROM mysql.gtid_slave_pos_InnoDB;
DROP TABLE mysql.gtid_slave_pos_InnoDB; DROP TABLE mysql.gtid_slave_pos_InnoDB;
SET sql_log_bin=1; SET sql_log_bin=1;
--write_file $MYSQLTEST_VARDIR/tmp/mysqld.2.expect --write_line "wait" $MYSQLTEST_VARDIR/tmp/mysqld.2.expect
wait
EOF
--shutdown_server --shutdown_server
--source include/wait_until_disconnected.inc --source include/wait_until_disconnected.inc
...@@ -272,9 +264,7 @@ while (!$done) ...@@ -272,9 +264,7 @@ while (!$done)
# MDEV-15373 engine gtid_slave_pos table name disobeys lower-case-table-names # MDEV-15373 engine gtid_slave_pos table name disobeys lower-case-table-names
# This snippet verifies that engine gtid_slave_pos table is found, # This snippet verifies that engine gtid_slave_pos table is found,
# its data are up-to-date. # its data are up-to-date.
--write_file $MYSQLTEST_VARDIR/tmp/mysqld.2.expect --write_line wait $MYSQLTEST_VARDIR/tmp/mysqld.2.expect
wait
EOF
--connection server_2 --connection server_2
--shutdown_server --shutdown_server
--source include/wait_until_disconnected.inc --source include/wait_until_disconnected.inc
......
...@@ -32,6 +32,10 @@ ...@@ -32,6 +32,10 @@
# MDEV-25616: Binlog event for XA COMMIT is generated without matching # MDEV-25616: Binlog event for XA COMMIT is generated without matching
# XA START, replication aborts # XA START, replication aborts
# #
# MDEV-33921: Replication fails when XA transactions are used where the slave
# has replicate_do_db set and the client has touched a different
# database when running DML such as inserts.
#
--source include/have_log_bin.inc --source include/have_log_bin.inc
--let $rpl_server_count= 3 --let $rpl_server_count= 3
...@@ -167,6 +171,129 @@ set @@global.binlog_format = row; ...@@ -167,6 +171,129 @@ set @@global.binlog_format = row;
set @@binlog_format = @sav_binlog_format; set @@binlog_format = @sav_binlog_format;
set @@global.binlog_format = @sav_binlog_format; set @@global.binlog_format = @sav_binlog_format;
--echo #
--echo # MDEV-33921.1: If a slave's replication of an XA transaction results in
--echo # an empty transaction, e.g. due to replication filters, the slave
--echo # should not binlog any part of the XA transaction.
#
# Note that the MDEV-33921 report is actually about that XA END is filtered
# out (not executed), and then its corresponding XA PREPARE errors because the
# XA state of the transaction is incorrect. This test case inherently tests
# both bugs.
--connection server_1
create database db1;
create database db2;
create table db1.t1 (a int) engine=innodb;
--source include/save_master_gtid.inc
--connection server_3
--source include/sync_with_master_gtid.inc
--source include/stop_slave.inc
--connection server_2
--source include/stop_slave.inc
SET @@GLOBAL.replicate_ignore_db= "";
SET @@GLOBAL.replicate_do_db= "db2";
--source include/start_slave.inc
--connection server_1
--let $pre_xa_gtid= `SELECT @@global.gtid_binlog_pos`
use db1;
XA START "x1";
insert into db1.t1 values (1);
XA END "x1";
XA PREPARE "x1";
XA COMMIT "x1";
--source include/save_master_gtid.inc
--connection server_2
--source include/sync_with_master_gtid.inc
--let $slave_binlogged_gtid= `SELECT @@global.gtid_binlog_pos`
if (`SELECT strcmp("$slave_binlogged_gtid","$pre_xa_gtid")`)
{
--die Slave binlogged an empty XA transaction yet should not have
}
--connection server_2
--source include/save_master_gtid.inc
--connection server_3
--source include/start_slave.inc
--source include/sync_with_master_gtid.inc
--echo #
--echo # 33921.2: If the slave shuts down after "preparing" a filtered-to-empty
--echo # XA transaction (and not completing it), then when the respective
--echo # XA completion (COMMIT in this test) command is replicated, the slave
--echo # should not throw ER_XAER_NOTA. Note that internally, the error is
--echo # thrown, but it is ignored because the target db is filtered.
--connection server_3
--source include/stop_slave.inc
--connection server_1
--let $pre_xa_gtid= `SELECT @@global.gtid_binlog_pos`
# Used by mysqlbinlog in part 3
--let $pre_xa_pos = query_get_value(SHOW MASTER STATUS, Position, 1)
use db1;
XA START "x2";
insert into db1.t1 values (2);
XA END "x2";
XA PREPARE "x2";
--source include/save_master_gtid.inc
--connection server_2
--source include/sync_with_master_gtid.inc
--let $rpl_server_number= 2
--echo # Connection named slave is needed for reconnection
--connect(slave,localhost,root,,)
--connect(slave1,localhost,root,,)
--source include/rpl_restart_server.inc
--connection server_2
--source include/stop_slave.inc
SET @@GLOBAL.replicate_do_db= "db2";
--source include/start_slave.inc
--connection server_1
XA COMMIT "x2";
--connection server_2
--source include/sync_with_master_gtid.inc
--source include/save_master_gtid.inc
--connection server_3
--source include/start_slave.inc
--source include/sync_with_master_gtid.inc
--echo #
--echo # 33921.3: Ensure XA commands are not considered by mysqlbinlog's
--echo # --database filter
--connection server_1
--let $datadir= `select @@datadir`
--let $binlog_file= query_get_value(SHOW MASTER STATUS, File, 1)
--let assert_file= $MYSQLTEST_VARDIR/tmp/binlog_decoded.out
--echo # MYSQL_BINLOG datadir/binlog_file --start-position=pre_xa_pos --database=db2 --result-file=assert_file
--exec $MYSQL_BINLOG $datadir/$binlog_file --start-position=$pre_xa_pos --database=db2 --result-file=$assert_file
--let assert_text= Mysqlbinlog should output all XA commands from the filtered transaction
--let assert_count= 4
--let assert_select= XA START|XA END|XA PREPARE|XA COMMIT
--source include/assert_grep.inc
--connection server_2
--source include/stop_slave.inc
SET @@GLOBAL.replicate_do_db="";
--source include/start_slave.inc
--connection server_1
drop database db1;
drop database db2;
# #
# Cleanup # Cleanup
--connection server_1 --connection server_1
......
...@@ -42,7 +42,7 @@ class Item_func_sysconst_test :public Item_func_sysconst ...@@ -42,7 +42,7 @@ class Item_func_sysconst_test :public Item_func_sysconst
} }
const char *fully_qualified_func_name() const override const char *fully_qualified_func_name() const override
{ return "sysconst_test()"; } { return "sysconst_test()"; }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_sysconst_test>(thd, this); } { return get_item_copy<Item_func_sysconst_test>(thd, this); }
}; };
......
...@@ -45,7 +45,7 @@ class Item_func_inet_aton : public Item_longlong_func ...@@ -45,7 +45,7 @@ class Item_func_inet_aton : public Item_longlong_func
unsigned_flag= 1; unsigned_flag= 1;
return FALSE; return FALSE;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_inet_aton>(thd, this); } { return get_item_copy<Item_func_inet_aton>(thd, this); }
}; };
...@@ -72,7 +72,7 @@ class Item_func_inet_ntoa : public Item_str_func ...@@ -72,7 +72,7 @@ class Item_func_inet_ntoa : public Item_str_func
set_maybe_null(); set_maybe_null();
return FALSE; return FALSE;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_inet_ntoa>(thd, this); } { return get_item_copy<Item_func_inet_ntoa>(thd, this); }
}; };
...@@ -118,7 +118,7 @@ class Item_func_inet6_aton : public Item_str_func ...@@ -118,7 +118,7 @@ class Item_func_inet6_aton : public Item_str_func
set_maybe_null(); set_maybe_null();
return FALSE; return FALSE;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_inet6_aton>(thd, this); } { return get_item_copy<Item_func_inet6_aton>(thd, this); }
String *val_str(String *to) override; String *val_str(String *to) override;
...@@ -156,7 +156,7 @@ class Item_func_inet6_ntoa : public Item_str_ascii_func ...@@ -156,7 +156,7 @@ class Item_func_inet6_ntoa : public Item_str_ascii_func
return FALSE; return FALSE;
} }
String *val_str_ascii(String *to) override; String *val_str_ascii(String *to) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_inet6_ntoa>(thd, this); } { return get_item_copy<Item_func_inet6_ntoa>(thd, this); }
}; };
...@@ -178,7 +178,7 @@ class Item_func_is_ipv4 : public Item_func_inet_bool_base ...@@ -178,7 +178,7 @@ class Item_func_is_ipv4 : public Item_func_inet_bool_base
static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4") }; static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_is_ipv4>(thd, this); } { return get_item_copy<Item_func_is_ipv4>(thd, this); }
longlong val_int() override; longlong val_int() override;
...@@ -201,7 +201,7 @@ class Item_func_is_ipv6 : public Item_func_inet_bool_base ...@@ -201,7 +201,7 @@ class Item_func_is_ipv6 : public Item_func_inet_bool_base
static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv6") }; static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv6") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_is_ipv6>(thd, this); } { return get_item_copy<Item_func_is_ipv6>(thd, this); }
longlong val_int() override; longlong val_int() override;
...@@ -223,7 +223,7 @@ class Item_func_is_ipv4_compat : public Item_func_inet_bool_base ...@@ -223,7 +223,7 @@ class Item_func_is_ipv4_compat : public Item_func_inet_bool_base
static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4_compat") }; static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4_compat") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_is_ipv4_compat>(thd, this); } { return get_item_copy<Item_func_is_ipv4_compat>(thd, this); }
longlong val_int() override; longlong val_int() override;
}; };
...@@ -244,7 +244,7 @@ class Item_func_is_ipv4_mapped : public Item_func_inet_bool_base ...@@ -244,7 +244,7 @@ class Item_func_is_ipv4_mapped : public Item_func_inet_bool_base
static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4_mapped") }; static LEX_CSTRING name= {STRING_WITH_LEN("is_ipv4_mapped") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_is_ipv4_mapped>(thd, this); } { return get_item_copy<Item_func_is_ipv4_mapped>(thd, this); }
longlong val_int() override; longlong val_int() override;
}; };
......
...@@ -1523,6 +1523,22 @@ int ha_prepare(THD *thd) ...@@ -1523,6 +1523,22 @@ int ha_prepare(THD *thd)
error=1; error=1;
} }
} }
else if (thd->rgi_slave)
{
/*
Slave threads will always process XA COMMITs in the binlog handler (see
MDEV-25616 and MDEV-30423), so if this is a slave thread preparing a
transaction which proved empty during replication (e.g. because of
replication filters) then mark it as XA_ROLLBACK_ONLY so the follow up
XA COMMIT will know to roll it back, rather than try to commit and binlog
a standalone XA COMMIT (without its preceding XA START - XA PREPARE).
If the xid_cache is cleared before the completion event comes, before
issuing ER_XAER_NOTA, first check if the event targets an ignored
database, and ignore the error if so.
*/
thd->transaction->xid_state.set_rollback_only();
}
DBUG_RETURN(error); DBUG_RETURN(error);
} }
......
...@@ -1284,6 +1284,25 @@ bool Item::eq(const Item *item, bool binary_cmp) const ...@@ -1284,6 +1284,25 @@ bool Item::eq(const Item *item, bool binary_cmp) const
} }
Item *Item::multiple_equality_transformer(THD *thd, uchar *arg)
{
if (const_item())
{
/*
Mark constant item in the condition with the MARKER_IMMUTABLE flag.
It is needed to prevent cleanup of the sub-items of this item and following
fix_fields() call that can cause a crash on this step of the optimization.
This flag will be removed at the end of the pushdown optimization by
remove_immutable_flag_processor processor.
*/
int new_flag= MARKER_IMMUTABLE;
this->walk(&Item::set_extraction_flag_processor, false,
(void*)&new_flag);
}
return this;
}
Item *Item::safe_charset_converter(THD *thd, CHARSET_INFO *tocs) Item *Item::safe_charset_converter(THD *thd, CHARSET_INFO *tocs)
{ {
if (!needs_charset_converter(tocs)) if (!needs_charset_converter(tocs))
...@@ -2753,7 +2772,7 @@ bool Type_std_attributes::agg_item_set_converter(const DTCollation &coll, ...@@ -2753,7 +2772,7 @@ bool Type_std_attributes::agg_item_set_converter(const DTCollation &coll,
@retval 0 on a failure @retval 0 on a failure
*/ */
Item* Item_func_or_sum::build_clone(THD *thd) Item* Item_func_or_sum::do_build_clone(THD *thd) const
{ {
Item *copy_tmp_args[2]= {0,0}; Item *copy_tmp_args[2]= {0,0};
Item **copy_args= copy_tmp_args; Item **copy_args= copy_tmp_args;
...@@ -3073,7 +3092,7 @@ Item_sp::init_result_field(THD *thd, uint max_length, uint maybe_null, ...@@ -3073,7 +3092,7 @@ Item_sp::init_result_field(THD *thd, uint max_length, uint maybe_null,
0 if an error occurred 0 if an error occurred
*/ */
Item* Item_ref::build_clone(THD *thd) Item* Item_ref::do_build_clone(THD *thd) const
{ {
Item_ref *copy= (Item_ref *) get_copy(thd); Item_ref *copy= (Item_ref *) get_copy(thd);
if (unlikely(!copy) || if (unlikely(!copy) ||
...@@ -3886,7 +3905,7 @@ void Item_decimal::set_decimal_value(my_decimal *value_par) ...@@ -3886,7 +3905,7 @@ void Item_decimal::set_decimal_value(my_decimal *value_par)
} }
Item *Item_decimal::clone_item(THD *thd) Item *Item_decimal::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_decimal(thd, name.str, &decimal_value, decimals, return new (thd->mem_root) Item_decimal(thd, name.str, &decimal_value, decimals,
max_length); max_length);
...@@ -3907,7 +3926,7 @@ my_decimal *Item_float::val_decimal(my_decimal *decimal_value) ...@@ -3907,7 +3926,7 @@ my_decimal *Item_float::val_decimal(my_decimal *decimal_value)
} }
Item *Item_float::clone_item(THD *thd) Item *Item_float::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_float(thd, name.str, value, decimals, return new (thd->mem_root) Item_float(thd, name.str, value, decimals,
max_length); max_length);
...@@ -4071,7 +4090,7 @@ Item *Item_null::safe_charset_converter(THD *thd, CHARSET_INFO *tocs) ...@@ -4071,7 +4090,7 @@ Item *Item_null::safe_charset_converter(THD *thd, CHARSET_INFO *tocs)
return this; return this;
} }
Item *Item_null::clone_item(THD *thd) Item *Item_null::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_null(thd, name.str); return new (thd->mem_root) Item_null(thd, name.str);
} }
...@@ -4877,7 +4896,7 @@ bool Item_param::basic_const_item() const ...@@ -4877,7 +4896,7 @@ bool Item_param::basic_const_item() const
} }
Item *Item_param::value_clone_item(THD *thd) Item *Item_param::value_clone_item(THD *thd) const
{ {
MEM_ROOT *mem_root= thd->mem_root; MEM_ROOT *mem_root= thd->mem_root;
switch (value.type_handler()->cmp_type()) { switch (value.type_handler()->cmp_type()) {
...@@ -4891,12 +4910,15 @@ Item *Item_param::value_clone_item(THD *thd) ...@@ -4891,12 +4910,15 @@ Item *Item_param::value_clone_item(THD *thd)
case DECIMAL_RESULT: case DECIMAL_RESULT:
return 0; // Should create Item_decimal. See MDEV-11361. return 0; // Should create Item_decimal. See MDEV-11361.
case STRING_RESULT: case STRING_RESULT:
{
String value_copy = value.m_string; // to preserve constness of the func
return new (mem_root) Item_string(thd, name, return new (mem_root) Item_string(thd, name,
Lex_cstring(value.m_string.ptr(), Lex_cstring(value_copy.ptr(),
value.m_string.length()), value_copy.length()),
value.m_string.charset(), value_copy.charset(),
collation.derivation, collation.derivation,
collation.repertoire); collation.repertoire);
}
case TIME_RESULT: case TIME_RESULT:
break; break;
case ROW_RESULT: case ROW_RESULT:
...@@ -4910,7 +4932,7 @@ Item *Item_param::value_clone_item(THD *thd) ...@@ -4910,7 +4932,7 @@ Item *Item_param::value_clone_item(THD *thd)
/* see comments in the header file */ /* see comments in the header file */
Item * Item *
Item_param::clone_item(THD *thd) Item_param::do_clone_const_item(THD *thd) const
{ {
// There's no "default". See comments in Item_param::save_in_field(). // There's no "default". See comments in Item_param::save_in_field().
switch (state) { switch (state) {
...@@ -7000,7 +7022,7 @@ int Item_string::save_in_field(Field *field, bool no_conversions) ...@@ -7000,7 +7022,7 @@ int Item_string::save_in_field(Field *field, bool no_conversions)
} }
Item *Item_string::clone_item(THD *thd) Item *Item_string::do_clone_const_item(THD *thd) const
{ {
LEX_CSTRING val; LEX_CSTRING val;
str_value.get_value(&val); str_value.get_value(&val);
...@@ -7064,7 +7086,7 @@ int Item_int::save_in_field(Field *field, bool no_conversions) ...@@ -7064,7 +7086,7 @@ int Item_int::save_in_field(Field *field, bool no_conversions)
} }
Item *Item_int::clone_item(THD *thd) Item *Item_int::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_int(thd, name.str, value, max_length, unsigned_flag); return new (thd->mem_root) Item_int(thd, name.str, value, max_length, unsigned_flag);
} }
...@@ -7093,7 +7115,7 @@ int Item_decimal::save_in_field(Field *field, bool no_conversions) ...@@ -7093,7 +7115,7 @@ int Item_decimal::save_in_field(Field *field, bool no_conversions)
} }
Item *Item_int_with_ref::clone_item(THD *thd) Item *Item_int_with_ref::do_clone_const_item(THD *thd) const
{ {
DBUG_ASSERT(ref->const_item()); DBUG_ASSERT(ref->const_item());
/* /*
...@@ -7189,7 +7211,7 @@ Item *Item_uint::neg(THD *thd) ...@@ -7189,7 +7211,7 @@ Item *Item_uint::neg(THD *thd)
} }
Item *Item_uint::clone_item(THD *thd) Item *Item_uint::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_uint(thd, name.str, value, max_length); return new (thd->mem_root) Item_uint(thd, name.str, value, max_length);
} }
...@@ -7429,7 +7451,7 @@ void Item_date_literal::print(String *str, enum_query_type query_type) ...@@ -7429,7 +7451,7 @@ void Item_date_literal::print(String *str, enum_query_type query_type)
} }
Item *Item_date_literal::clone_item(THD *thd) Item *Item_date_literal::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_date_literal(thd, &cached_time); return new (thd->mem_root) Item_date_literal(thd, &cached_time);
} }
...@@ -7454,7 +7476,7 @@ void Item_datetime_literal::print(String *str, enum_query_type query_type) ...@@ -7454,7 +7476,7 @@ void Item_datetime_literal::print(String *str, enum_query_type query_type)
} }
Item *Item_datetime_literal::clone_item(THD *thd) Item *Item_datetime_literal::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_datetime_literal(thd, &cached_time, decimals); return new (thd->mem_root) Item_datetime_literal(thd, &cached_time, decimals);
} }
...@@ -7479,7 +7501,7 @@ void Item_time_literal::print(String *str, enum_query_type query_type) ...@@ -7479,7 +7501,7 @@ void Item_time_literal::print(String *str, enum_query_type query_type)
} }
Item *Item_time_literal::clone_item(THD *thd) Item *Item_time_literal::do_clone_const_item(THD *thd) const
{ {
return new (thd->mem_root) Item_time_literal(thd, &cached_time, decimals); return new (thd->mem_root) Item_time_literal(thd, &cached_time, decimals);
} }
...@@ -10435,7 +10457,7 @@ void Item_cache_temporal::store_packed(longlong val_arg, Item *example_arg) ...@@ -10435,7 +10457,7 @@ void Item_cache_temporal::store_packed(longlong val_arg, Item *example_arg)
} }
Item *Item_cache_temporal::clone_item(THD *thd) Item *Item_cache_temporal::do_clone_const_item(THD *thd) const
{ {
Item_cache *tmp= type_handler()->Item_get_cache(thd, this); Item_cache *tmp= type_handler()->Item_get_cache(thd, this);
Item_cache_temporal *item= static_cast<Item_cache_temporal*>(tmp); Item_cache_temporal *item= static_cast<Item_cache_temporal*>(tmp);
......
This diff is collapsed.
...@@ -5546,17 +5546,16 @@ void Item_cond::neg_arguments(THD *thd) ...@@ -5546,17 +5546,16 @@ void Item_cond::neg_arguments(THD *thd)
0 if an error occurred 0 if an error occurred
*/ */
Item *Item_cond::build_clone(THD *thd) Item *Item_cond::do_build_clone(THD *thd) const
{ {
List_iterator_fast<Item> li(list);
Item *item;
Item_cond *copy= (Item_cond *) get_copy(thd); Item_cond *copy= (Item_cond *) get_copy(thd);
if (!copy) if (!copy)
return 0; return 0;
copy->list.empty(); copy->list.empty();
while ((item= li++))
for (const Item &item : list)
{ {
Item *arg_clone= item->build_clone(thd); Item *arg_clone= item.build_clone(thd);
if (!arg_clone) if (!arg_clone)
return 0; return 0;
if (copy->list.push_back(arg_clone, thd->mem_root)) if (copy->list.push_back(arg_clone, thd->mem_root))
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -101,7 +101,7 @@ class Item_func_json_valid: public Item_bool_func ...@@ -101,7 +101,7 @@ class Item_func_json_valid: public Item_bool_func
static const Lex_cstring fmt(STRING_WITH_LEN("json")); static const Lex_cstring fmt(STRING_WITH_LEN("json"));
return to->set_format_name(fmt); return to->set_format_name(fmt);
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_valid>(thd, this); } { return get_item_copy<Item_func_json_valid>(thd, this); }
enum Functype functype() const override { return JSON_VALID_FUNC; } enum Functype functype() const override { return JSON_VALID_FUNC; }
}; };
...@@ -122,7 +122,7 @@ class Item_func_json_exists: public Item_bool_func ...@@ -122,7 +122,7 @@ class Item_func_json_exists: public Item_bool_func
return name; return name;
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_exists>(thd, this); } { return get_item_copy<Item_func_json_exists>(thd, this); }
longlong val_int() override; longlong val_int() override;
}; };
...@@ -170,7 +170,7 @@ class Item_func_json_value: public Item_str_func, ...@@ -170,7 +170,7 @@ class Item_func_json_value: public Item_str_func,
{ {
return je->check_and_get_value_scalar(res, error); return je->check_and_get_value_scalar(res, error);
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_value>(thd, this); } { return get_item_copy<Item_func_json_value>(thd, this); }
}; };
...@@ -198,7 +198,7 @@ class Item_func_json_query: public Item_json_func, ...@@ -198,7 +198,7 @@ class Item_func_json_query: public Item_json_func,
{ {
return je->check_and_get_value_complex(res, error); return je->check_and_get_value_complex(res, error);
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_query>(thd, this); } { return get_item_copy<Item_func_json_query>(thd, this); }
}; };
...@@ -217,7 +217,7 @@ class Item_func_json_quote: public Item_str_func ...@@ -217,7 +217,7 @@ class Item_func_json_quote: public Item_str_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *) override; String *val_str(String *) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_quote>(thd, this); } { return get_item_copy<Item_func_json_quote>(thd, this); }
}; };
...@@ -236,7 +236,7 @@ class Item_func_json_unquote: public Item_str_func ...@@ -236,7 +236,7 @@ class Item_func_json_unquote: public Item_str_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *) override; String *val_str(String *) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_unquote>(thd, this); } { return get_item_copy<Item_func_json_unquote>(thd, this); }
}; };
...@@ -291,7 +291,7 @@ class Item_func_json_extract: public Item_json_str_multipath ...@@ -291,7 +291,7 @@ class Item_func_json_extract: public Item_json_str_multipath
double val_real() override; double val_real() override;
my_decimal *val_decimal(my_decimal *) override; my_decimal *val_decimal(my_decimal *) override;
uint get_n_paths() const override { return arg_count - 1; } uint get_n_paths() const override { return arg_count - 1; }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_extract>(thd, this); } { return get_item_copy<Item_func_json_extract>(thd, this); }
}; };
...@@ -314,7 +314,7 @@ class Item_func_json_contains: public Item_bool_func ...@@ -314,7 +314,7 @@ class Item_func_json_contains: public Item_bool_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
longlong val_int() override; longlong val_int() override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_contains>(thd, this); } { return get_item_copy<Item_func_json_contains>(thd, this); }
}; };
...@@ -341,7 +341,7 @@ class Item_func_json_contains_path: public Item_bool_func ...@@ -341,7 +341,7 @@ class Item_func_json_contains_path: public Item_bool_func
bool fix_fields(THD *thd, Item **ref) override; bool fix_fields(THD *thd, Item **ref) override;
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
longlong val_int() override; longlong val_int() override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_contains_path>(thd, this); } { return get_item_copy<Item_func_json_contains_path>(thd, this); }
}; };
...@@ -363,7 +363,7 @@ class Item_func_json_array: public Item_json_func ...@@ -363,7 +363,7 @@ class Item_func_json_array: public Item_json_func
static LEX_CSTRING name= {STRING_WITH_LEN("json_array") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_array") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_array>(thd, this); } { return get_item_copy<Item_func_json_array>(thd, this); }
}; };
...@@ -384,7 +384,7 @@ class Item_func_json_array_append: public Item_json_str_multipath ...@@ -384,7 +384,7 @@ class Item_func_json_array_append: public Item_json_str_multipath
static LEX_CSTRING name= {STRING_WITH_LEN("json_array_append") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_array_append") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_array_append>(thd, this); } { return get_item_copy<Item_func_json_array_append>(thd, this); }
}; };
...@@ -400,7 +400,7 @@ class Item_func_json_array_insert: public Item_func_json_array_append ...@@ -400,7 +400,7 @@ class Item_func_json_array_insert: public Item_func_json_array_append
static LEX_CSTRING name= {STRING_WITH_LEN("json_array_insert") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_array_insert") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_array_insert>(thd, this); } { return get_item_copy<Item_func_json_array_insert>(thd, this); }
}; };
...@@ -418,7 +418,7 @@ class Item_func_json_object: public Item_func_json_array ...@@ -418,7 +418,7 @@ class Item_func_json_object: public Item_func_json_array
static LEX_CSTRING name= {STRING_WITH_LEN("json_object") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_object") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_object>(thd, this); } { return get_item_copy<Item_func_json_object>(thd, this); }
}; };
...@@ -436,7 +436,7 @@ class Item_func_json_merge: public Item_func_json_array ...@@ -436,7 +436,7 @@ class Item_func_json_merge: public Item_func_json_array
static LEX_CSTRING name= {STRING_WITH_LEN("json_merge_preserve") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_merge_preserve") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_merge>(thd, this); } { return get_item_copy<Item_func_json_merge>(thd, this); }
}; };
...@@ -451,7 +451,7 @@ class Item_func_json_merge_patch: public Item_func_json_merge ...@@ -451,7 +451,7 @@ class Item_func_json_merge_patch: public Item_func_json_merge
return name; return name;
} }
String *val_str(String *) override; String *val_str(String *) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_merge_patch>(thd, this); } { return get_item_copy<Item_func_json_merge_patch>(thd, this); }
}; };
...@@ -482,7 +482,7 @@ class Item_func_json_length: public Item_long_func ...@@ -482,7 +482,7 @@ class Item_func_json_length: public Item_long_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
longlong val_int() override; longlong val_int() override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_length>(thd, this); } { return get_item_copy<Item_func_json_length>(thd, this); }
}; };
...@@ -502,7 +502,7 @@ class Item_func_json_depth: public Item_long_func ...@@ -502,7 +502,7 @@ class Item_func_json_depth: public Item_long_func
} }
bool fix_length_and_dec() override { max_length= 10; return FALSE; } bool fix_length_and_dec() override { max_length= 10; return FALSE; }
longlong val_int() override; longlong val_int() override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_depth>(thd, this); } { return get_item_copy<Item_func_json_depth>(thd, this); }
}; };
...@@ -520,7 +520,7 @@ class Item_func_json_type: public Item_str_func ...@@ -520,7 +520,7 @@ class Item_func_json_type: public Item_str_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *) override; String *val_str(String *) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_type>(thd, this); } { return get_item_copy<Item_func_json_type>(thd, this); }
}; };
...@@ -546,7 +546,7 @@ class Item_func_json_insert: public Item_json_str_multipath ...@@ -546,7 +546,7 @@ class Item_func_json_insert: public Item_json_str_multipath
return (mode_insert ? return (mode_insert ?
(mode_replace ? json_set : json_insert) : json_replace); (mode_replace ? json_set : json_insert) : json_replace);
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_insert>(thd, this); } { return get_item_copy<Item_func_json_insert>(thd, this); }
}; };
...@@ -566,7 +566,7 @@ class Item_func_json_remove: public Item_json_str_multipath ...@@ -566,7 +566,7 @@ class Item_func_json_remove: public Item_json_str_multipath
static LEX_CSTRING name= {STRING_WITH_LEN("json_remove") }; static LEX_CSTRING name= {STRING_WITH_LEN("json_remove") };
return name; return name;
} }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_remove>(thd, this); } { return get_item_copy<Item_func_json_remove>(thd, this); }
}; };
...@@ -587,7 +587,7 @@ class Item_func_json_keys: public Item_str_func ...@@ -587,7 +587,7 @@ class Item_func_json_keys: public Item_str_func
} }
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *) override; String *val_str(String *) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_keys>(thd, this); } { return get_item_copy<Item_func_json_keys>(thd, this); }
}; };
...@@ -616,7 +616,7 @@ class Item_func_json_search: public Item_json_str_multipath ...@@ -616,7 +616,7 @@ class Item_func_json_search: public Item_json_str_multipath
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *) override; String *val_str(String *) override;
uint get_n_paths() const override { return arg_count > 4 ? arg_count - 4 : 0; } uint get_n_paths() const override { return arg_count > 4 ? arg_count - 4 : 0; }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_search>(thd, this); } { return get_item_copy<Item_func_json_search>(thd, this); }
}; };
...@@ -644,7 +644,7 @@ class Item_func_json_format: public Item_json_func ...@@ -644,7 +644,7 @@ class Item_func_json_format: public Item_json_func
bool fix_length_and_dec() override; bool fix_length_and_dec() override;
String *val_str(String *str) override; String *val_str(String *str) override;
String *val_json(String *str) override; String *val_json(String *str) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_format>(thd, this); } { return get_item_copy<Item_func_json_format>(thd, this); }
}; };
...@@ -690,7 +690,7 @@ class Item_func_json_arrayagg : public Item_func_group_concat ...@@ -690,7 +690,7 @@ class Item_func_json_arrayagg : public Item_func_group_concat
String* val_str(String *str) override; String* val_str(String *str) override;
Item *copy_or_same(THD* thd) override; Item *copy_or_same(THD* thd) override;
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_arrayagg>(thd, this); } { return get_item_copy<Item_func_json_arrayagg>(thd, this); }
}; };
...@@ -739,7 +739,7 @@ class Item_func_json_objectagg : public Item_sum ...@@ -739,7 +739,7 @@ class Item_func_json_objectagg : public Item_sum
String* val_str(String* str) override; String* val_str(String* str) override;
Item *copy_or_same(THD* thd) override; Item *copy_or_same(THD* thd) override;
void no_rows_in_result() override {} void no_rows_in_result() override {}
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_func_json_objectagg>(thd, this); } { return get_item_copy<Item_func_json_objectagg>(thd, this); }
}; };
......
...@@ -180,7 +180,7 @@ void Item_row::bring_value() ...@@ -180,7 +180,7 @@ void Item_row::bring_value()
} }
Item* Item_row::build_clone(THD *thd) Item* Item_row::do_build_clone(THD *thd) const
{ {
Item **copy_args= static_cast<Item**> Item **copy_args= static_cast<Item**>
(alloc_root(thd->mem_root, sizeof(Item*) * arg_count)); (alloc_root(thd->mem_root, sizeof(Item*) * arg_count));
......
...@@ -148,9 +148,9 @@ class Item_row: public Item_fixed_hybrid, ...@@ -148,9 +148,9 @@ class Item_row: public Item_fixed_hybrid,
} }
bool check_vcol_func_processor(void *arg) override {return FALSE; } bool check_vcol_func_processor(void *arg) override {return FALSE; }
Item *get_copy(THD *thd) override Item *do_get_copy(THD *thd) const override
{ return get_item_copy<Item_row>(thd, this); } { return get_item_copy<Item_row>(thd, this); }
Item *build_clone(THD *thd) override; Item *do_build_clone(THD *thd) const override;
}; };
#endif /* ITEM_ROW_INCLUDED */ #endif /* ITEM_ROW_INCLUDED */
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -56,6 +56,7 @@ class Rpl_filter ...@@ -56,6 +56,7 @@ class Rpl_filter
bool db_ok_with_wild_table(const char *db); bool db_ok_with_wild_table(const char *db);
bool is_on(); bool is_on();
bool is_db_empty();
/* Setters - add filtering rules */ /* Setters - add filtering rules */
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
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