Thank you!
I repeated as described on 10.4-11.3 with InnoDB, not Myisam.
On the release build it crashes after 20-30 executions of the query, while on debug on my machine that was ~9000 executions:
Query (0x62b0000a1290): SELECT f6 FROM (SELECT DISTINCT `c1` AS `f6` FROM `t1`) AS `t1`
JOIN (SELECT CHARSET(`c2`) AS `f5` FROM `t2`) AS `t3`
ON (EXISTS (SELECT `c2` FROM `t2`)) AND
(NOT ((CHARSET(`f5`)) NOT BETWEEN `f6` AND `f6`))
Alice Sherepa
added a comment - - edited Thank you!
I repeated as described on 10.4-11.3 with InnoDB, not Myisam.
On the release build it crashes after 20-30 executions of the query, while on debug on my machine that was ~9000 executions:
--source include/have_innodb.inc
create table t1 (`c1` bigint , key (`c1`)) engine=innodb;
insert into t1 values (-2);
create table t2 ( `c2` double unsigned );
insert into t2 values (12.991);
let $1= 10000;
while ($1)
{
SELECT f6 FROM ( SELECT DISTINCT `c1` AS `f6` FROM `t1`) AS `t1`
JOIN ( SELECT CHARSET(`c2`) AS `f5` FROM `t2`) AS `t3`
ON (EXISTS ( SELECT `c2` FROM `t2`)) AND
( NOT ((CHARSET(`f5`)) NOT BETWEEN `f6` AND `f6`));
dec $1;
}
drop table t1,t2;
231214 11:22:49 [ERROR] mysqld got signal 11 ;
Server version: 10.4.33-MariaDB-debug-log source revision: 9f5078a1d79031c4a781d378af18df9c8c9d2321
sql/signal_handler.cc:235(handle_fatal_signal)[0x558d689effd3]
sigaction.c:0(__restore_rt)[0x7f9a6f82c420]
sql/sql_select.cc:7139(optimize_keyuse(JOIN*, st_dynamic_array*))[0x558d6831ceba]
sql/sql_select.cc:5708(make_join_statistics(JOIN*, List<TABLE_LIST>&, st_dynamic_array*))[0x558d6830fb03]
sql/sql_select.cc:2388(JOIN::optimize_inner())[0x558d682ed791]
sql/sql_select.cc:1731(JOIN::optimize())[0x558d682e67a7]
sql/sql_derived.cc:1029(mysql_derived_optimize(THD*, LEX*, TABLE_LIST*))[0x558d681529ef]
sql/sql_derived.cc:200(mysql_handle_single_derived(LEX*, TABLE_LIST*, unsigned int))[0x558d6814d19a]
sql/sql_select.cc:2221(JOIN::optimize_inner())[0x558d682ebd0c]
sql/sql_select.cc:1731(JOIN::optimize())[0x558d682e67a7]
sql/sql_select.cc:4832(mysql_select(THD*, TABLE_LIST*, unsigned int, List<Item>&, Item*, unsigned int, st_order*, st_order*, Item*, st_order*, unsigned long long, select_result*, st_select_lex_unit*, st_select_lex*))[0x558d68307791]
sql/sql_select.cc:442(handle_select(THD*, LEX*, select_result*, unsigned long))[0x558d682d81f8]
sql/sql_parse.cc:6523(execute_sqlcom_select(THD*, TABLE_LIST*))[0x558d6823f1fd]
sql/sql_parse.cc:3980(mysql_execute_command(THD*))[0x558d6822c81d]
sql/sql_parse.cc:8062(mysql_parse(THD*, char*, unsigned int, Parser_state*, bool, bool))[0x558d68248779]
sql/sql_parse.cc:1860(dispatch_command(enum_server_command, THD*, char*, unsigned int, bool, bool))[0x558d6821e925]
sql/sql_parse.cc:1378(do_command(THD*))[0x558d6821b450]
sql/sql_connect.cc:1419(do_handle_one_connection(CONNECT*))[0x558d686304c6]
sql/sql_connect.cc:1324(handle_one_connection)[0x558d6862fd6a]
perfschema/pfs.cc:1871(pfs_spawn_thread)[0x558d692cde4c]
nptl/pthread_create.c:478(start_thread)[0x7f9a6f820609]
Query (0x62b0000a1290): SELECT f6 FROM (SELECT DISTINCT `c1` AS `f6` FROM `t1`) AS `t1`
JOIN (SELECT CHARSET(`c2`) AS `f5` FROM `t2`) AS `t3`
ON (EXISTS (SELECT `c2` FROM `t2`)) AND
(NOT ((CHARSET(`f5`)) NOT BETWEEN `f6` AND `f6`))
Below is the explain plan obtained from version 10.5:
EXPLAIN EXTENDED
SELECT f6 FROM (SELECT DISTINCT `c1` AS `f6` FROM `t1`) AS `dt1`
JOIN (SELECT CHARSET(`c2`) AS `f5` FROM `t2`) AS `dt3`
ON (EXISTS (SELECT `c2` FROM `t2`)) AND
(NOT ((CHARSET(`f5`)) NOT BETWEEN `f6` AND `f6`));
id select_type table type possible_keys key key_len ref rows filtered Extra
1 PRIMARY <derived3> system NULL NULL NULL NULL 1 100.00
1 PRIMARY <derived2> ref key0 key0 9 const 0 0.00 Using where
4 SUBQUERY t2 system NULL NULL NULL NULL 1 100.00
3 DERIVED t2 system NULL NULL NULL NULL 1 100.00
2 DERIVED t1 range c1 c1 9 NULL 2 50.00 Using where; Using index for group-by
Warnings:
Note 1003 /* select#1 */ select `dt1`.`f6` AS `f6` from (/* select#2 */ select distinct `test`.`t1`.`c1` AS `f6` from `test`.`t1` where <cache>(charset('binary')) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1` where <cache>(charset('binary')) between `dt1`.`f6` and `dt1`.`f6`
The crash happens because keyuse->used_tables == 2 when JOIN::table_count == 1. keyuse->used_tables == 2 due to keyuse->val->used_tables() == 2 and keyuse->val is Item_func_charset which is dbug_print'ed as "charset(t3.f5)".
As I can see, there are two JOIN objects involved here, the inner one is initialized at mysql_derived_optimize (frame 5 of the call stack) as
The outer join processed at frame 7 has table_count == 2, so keyuse->val->used_tables() == 2 makes sense for that join.
Oleg Smirnov
added a comment - Below is the explain plan obtained from version 10.5:
EXPLAIN EXTENDED
SELECT f6 FROM (SELECT DISTINCT `c1` AS `f6` FROM `t1`) AS `dt1`
JOIN (SELECT CHARSET(`c2`) AS `f5` FROM `t2`) AS `dt3`
ON (EXISTS (SELECT `c2` FROM `t2`)) AND
(NOT ((CHARSET(`f5`)) NOT BETWEEN `f6` AND `f6`));
id select_type table type possible_keys key key_len ref rows filtered Extra
1 PRIMARY <derived3> system NULL NULL NULL NULL 1 100.00
1 PRIMARY <derived2> ref key0 key0 9 const 0 0.00 Using where
4 SUBQUERY t2 system NULL NULL NULL NULL 1 100.00
3 DERIVED t2 system NULL NULL NULL NULL 1 100.00
2 DERIVED t1 range c1 c1 9 NULL 2 50.00 Using where; Using index for group-by
Warnings:
Note 1003 /* select#1 */ select `dt1`.`f6` AS `f6` from (/* select#2 */ select distinct `test`.`t1`.`c1` AS `f6` from `test`.`t1` where <cache>(charset('binary')) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1` where <cache>(charset('binary')) between `dt1`.`f6` and `dt1`.`f6`
Call stack of the crash:
1 optimize_keyuse sql_select.cc 7189 0x555555f24f48
2 make_join_statistics sql_select.cc 5751 0x555555f2085e
3 JOIN::optimize_inner sql_select.cc 2386 0x555555f142f1
4 JOIN::optimize sql_select.cc 1740 0x555555f11c24
5 mysql_derived_optimize sql_derived.cc 1029 0x555555e60377
6 mysql_handle_single_derived sql_derived.cc 200 0x555555e5dd99
7 JOIN::optimize_inner sql_select.cc 2204 0x555555f1387d
8 JOIN::optimize sql_select.cc 1740 0x555555f11c24
9 mysql_select sql_select.cc 4868 0x555555f1d66f
10 mysql_explain_union sql_select.cc 28125 0x555555f5efa7
11 execute_sqlcom_select sql_parse.cc 6356 0x555555ecada9
12 mysql_execute_command sql_parse.cc 4022 0x555555ec1f97
... <More>
The crash happens because keyuse->used_tables == 2 when JOIN::table_count == 1 . keyuse->used_tables == 2 due to keyuse->val->used_tables() == 2 and keyuse->val is Item_func_charset which is dbug_print'ed as "charset(t3.f5)".
As I can see, there are two JOIN objects involved here, the inner one is initialized at mysql_derived_optimize (frame 5 of the call stack) as
JOIN *join= derived->get_unit()->first_select()->join;
The outer join processed at frame 7 has table_count == 2 , so keyuse->val->used_tables() == 2 makes sense for that join.
Simplified test case (crashes consistently, no need to run a cycle):
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON CHARSET(dt2_c2) BETWEEN dt1_c1 AND dt1_c1;
Warnings:
Note 1003
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`
where <cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1`
where <cache>(charset(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
Note: rewritten query text in the warning is obtained by disabling the optimize_keyuse() call.
<cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1` is shown twice in the output, doesn't it look strange?
Observation: replacing CHARSET() with LENGTH() eliminates the crash, and the execution doesn't have duplicate call of the function octet_length():
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON LENGTH(dt2_c2) BETWEEN dt1_c1 AND dt1_c1;
id select_type table type possible_keys key key_len ref rows filtered Extra
1 PRIMARY <derived3> system NULL NULL NULL NULL 1 100.00
1 PRIMARY <derived2> ref key0 key0 9 const 0 0.00 Using where
3 DERIVED t2 system NULL NULL NULL NULL 1 100.00
2 DERIVED t1 index NULL c1 9 NULL 1 100.00 Using index
Warnings:
Note 1003
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`) `dt1`
where <cache>(octet_length(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
Oleg Smirnov
added a comment - - edited Simplified test case (crashes consistently, no need to run a cycle):
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON CHARSET(dt2_c2) BETWEEN dt1_c1 AND dt1_c1;
Warnings:
Note 1003
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`
where <cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1`
where <cache>(charset(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
Note: rewritten query text in the warning is obtained by disabling the optimize_keyuse() call.
<cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1` is shown twice in the output, doesn't it look strange?
Observation: replacing CHARSET() with LENGTH() eliminates the crash, and the execution doesn't have duplicate call of the function octet_length() :
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON LENGTH(dt2_c2) BETWEEN dt1_c1 AND dt1_c1;
id select_type table type possible_keys key key_len ref rows filtered Extra
1 PRIMARY <derived3> system NULL NULL NULL NULL 1 100.00
1 PRIMARY <derived2> ref key0 key0 9 const 0 0.00 Using where
3 DERIVED t2 system NULL NULL NULL NULL 1 100.00
2 DERIVED t1 index NULL c1 9 NULL 1 100.00 Using index
Warnings:
Note 1003
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`) `dt1`
where <cache>(octet_length(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
Looking at the result of rewrite of the problematic query
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`
where <cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1`
where <cache>(charset(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
I can see that condition CHARSET(dt2_c2) BETWEEN dt1_c1 AND dt1_c1 has been pushed down into the derived table `dt1`, which must not have happened. Only conditions depending solely on the derived table can be pushed into, but CHARSET(dt2_c2) is dependent on `dt2` and must not be pushed down.
The pushdown logic checks table dependencies but skips the check for const items. Const items are always considered as eligible for pushdown, and in this case Item_func_charset turns out to be erroneously const.
This happens because Item_func_charset inherits from Item_func_expr_str_metadata which overrides const_item() in the following way, forcing the const property:
bool const_item() const { return true; }
That was introduced with the commit
commit aa1002a35cefcc1851cbfb6b6b60463bda6f9aa3
Author: Alexander Barkov <bar@mariadb.org>, Wed Sep 2 08:20:49 2015 +0400 (9 years ago)
MDEV-8723 Wrong result for SELECT..WHERE COLLATION(a)='binary' AND a='a'
The same was made for Item_func_coercibility, so the crash is reproducible if CHARSET() is replaced with COERCIBILITY().
Oleg Smirnov
added a comment - Looking at the result of rewrite of the problematic query
/* select#1 */ select `dt1`.`dt1_c1` AS `dt1_c1`
from
(/* select#2 */ select `test`.`t1`.`c1` AS `dt1_c1` from `test`.`t1`
where <cache>(charset(1)) between `test`.`t1`.`c1` and `test`.`t1`.`c1`) `dt1`
where <cache>(charset(1)) between `dt1`.`dt1_c1` and `dt1`.`dt1_c1`
I can see that condition CHARSET(dt2_c2) BETWEEN dt1_c1 AND dt1_c1 has been pushed down into the derived table `dt1`, which must not have happened. Only conditions depending solely on the derived table can be pushed into, but CHARSET(dt2_c2) is dependent on `dt2` and must not be pushed down.
The pushdown logic checks table dependencies but skips the check for const items. Const items are always considered as eligible for pushdown, and in this case Item_func_charset turns out to be erroneously const.
This happens because Item_func_charset inherits from Item_func_expr_str_metadata which overrides const_item() in the following way, forcing the const property:
bool const_item() const { return true; }
That was introduced with the commit
commit aa1002a35cefcc1851cbfb6b6b60463bda6f9aa3
Author: Alexander Barkov <bar@mariadb.org>, Wed Sep 2 08:20:49 2015 +0400 (9 years ago)
MDEV-8723 Wrong result for SELECT..WHERE COLLATION(a)='binary' AND a='a'
The same was made for Item_func_coercibility , so the crash is reproducible if CHARSET() is replaced with COERCIBILITY() .
bar , psergei , please review the pull request .
I think functions CHARSET() and COERCIBILITY should return const_item()==true. They are definitely constants during the query. They don't change their value per row.
They are very close to DEFAULT(field) in this sense. I checked this script:
createorreplacetable t1(a intdefault 1);
insertinto t1 values (1),(2);
select * from t1 where a=default(a);
and Item_default_value::const_item() returns true during the select statement.
Can you please check the reported problem with Item_default_value? Does it also crash?
Alexander Barkov
added a comment - Hi oleg.smirnov ,
I think functions CHARSET() and COERCIBILITY should return const_item()==true. They are definitely constants during the query. They don't change their value per row.
They are very close to DEFAULT(field) in this sense. I checked this script:
create or replace table t1(a int default 1);
insert into t1 values (1),(2);
select * from t1 where a= default (a);
and Item_default_value::const_item() returns true during the select statement.
Can you please check the reported problem with Item_default_value? Does it also crash?
I filed a separate issue for DEFAULT() 'cause there is another call stack and it's reproducible not only on InnoDB but on MyISAM as well.
Oleg Smirnov
added a comment - I filed a separate issue for DEFAULT() 'cause there is another call stack and it's reproducible not only on InnoDB but on MyISAM as well.
I agree with bar that CHARSET() and COERCIBILITY() should return const_item()==true. So this should be something related to the "condition pushdown into derived" optimization.
A little bit more simplified test case:
--source include/have_innodb.inc
CREATE TABLE t1 (c1 INT, KEY (c1)) ENGINE=InnoDB;
INSERT INTO t1 VALUES (1);
CREATE TABLE t2 (c2 INT);
INSERT INTO t2 VALUES (1);
SET optimizer_switch='derived_merge=off';
set optimizer_trace=1;
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON CHARSET(dt2_c2) = dt1_c1;
To reiterate: CHARSET(dt2_c2) = dt1_c1 is pushed down into dt1: SELECT c1 AS dt1_c1 FROM t1 WHERE CHARSET(dt2_c2) = dt1_c1 and CHARSET(dt2_c2) is a const item. The optimizer considers using KEY(c1) on t1 for value CHARSET(dt2_c2) and performs some operations at optimize_keyuse(). Since CHARSET(dt2_c2)::used_tables() == 2 and the dt1's JOIN::table_count == 1, a crash occurs:
The first idea: condition with CHARSET(dt2_c2) must not be pushed into dt1 (but why not if it's const?)
Second idea: shouldn't we have JOIN::const_table_map == 2 after the pushdown? That eliminates the crash at optimize_keyuse().
Oleg Smirnov
added a comment - I agree with bar that CHARSET() and COERCIBILITY() should return const_item()==true. So this should be something related to the "condition pushdown into derived" optimization.
A little bit more simplified test case:
--source include/have_innodb.inc
CREATE TABLE t1 (c1 INT, KEY (c1)) ENGINE=InnoDB;
INSERT INTO t1 VALUES (1);
CREATE TABLE t2 (c2 INT);
INSERT INTO t2 VALUES (1);
SET optimizer_switch='derived_merge=off';
set optimizer_trace=1;
EXPLAIN EXTENDED
SELECT dt1_c1 FROM
(SELECT c1 AS dt1_c1 FROM t1) AS dt1
JOIN
(SELECT 1 AS dt2_c2 FROM t2) AS dt2
ON CHARSET(dt2_c2) = dt1_c1;
To reiterate: CHARSET(dt2_c2) = dt1_c1 is pushed down into dt1 : SELECT c1 AS dt1_c1 FROM t1 WHERE CHARSET(dt2_c2) = dt1_c1 and CHARSET(dt2_c2) is a const item. The optimizer considers using KEY(c1) on t1 for value CHARSET(dt2_c2) and performs some operations at optimize_keyuse() . Since CHARSET(dt2_c2)::used_tables() == 2 and the dt1 's JOIN::table_count == 1, a crash occurs:
optimize_keyuse()
if (keyuse->used_tables &
(map= (keyuse->used_tables & ~join->const_table_map &
~OUTER_REF_TABLE_BIT)))
{
uint n_tables= my_count_bits(map);
if (n_tables == 1) // Only one table
{
DBUG_ASSERT(!(map & PSEUDO_TABLE_BITS)); // Must be a real table
Table_map_iterator it(map);
int tablenr= it.next_bit();
DBUG_ASSERT(tablenr != Table_map_iterator::BITMAP_END);
TABLE *tmp_table=join->table[tablenr];
if (tmp_table) // already created
keyuse->ref_table_rows= MY_MAX(tmp_table->file->stats.records, 100);
}
}
The first idea: condition with CHARSET(dt2_c2) must not be pushed into dt1 (but why not if it's const?)
Second idea: shouldn't we have JOIN::const_table_map == 2 after the pushdown? That eliminates the crash at optimize_keyuse().
The first idea: condition with CHARSET(dt2_c2) must not be pushed into dt1 (but why not if it's const?)
This may or may not be the solution. Will elaborate below.
(I'm wondering, why does pushdown code check const_tables and not just use used_tables ?)
Second idea: shouldn't we have JOIN::const_table_map == 2 after the pushdown? That eliminates the crash at optimize_keyuse().
Do you the child join object should have const_table_map=2? This doesn't look like a good idea, because the number 2 (table bit #1 counting from 0) comes from the parent select. The child join may not have two tables (like in this example) or it may have another table with table->map=2.
Sergei Petrunia
added a comment -
The first idea: condition with CHARSET(dt2_c2) must not be pushed into dt1 (but why not if it's const?)
This may or may not be the solution. Will elaborate below.
(I'm wondering, why does pushdown code check const_tables and not just use used_tables ?)
Second idea: shouldn't we have JOIN::const_table_map == 2 after the pushdown? That eliminates the crash at optimize_keyuse().
Do you the child join object should have const_table_map=2 ? This doesn't look like a good idea, because the number 2 (table bit #1 counting from 0) comes from the parent select. The child join may not have two tables (like in this example) or it may have another table with table->map=2.
The fact that CHARSET(dt2_c2)::used_tables() == 2 after this expression has been pushed into dt1 is wrong.
I can see two options:
If CHARSET(...) continues to refer to dt2, then it should have used_tables()=OUTER_REF_TABLE_BIT now. This is a bad approach to take as the optimizer doesn't currently support outer references inside derived tables.
Can we assume that CHARSET(dt2.dt2_c2) does not depend on dt2 and so it should have used_tables()=0?
Sergei Petrunia
added a comment - - edited The fact that CHARSET(dt2_c2)::used_tables() == 2 after this expression has been pushed into dt1 is wrong.
I can see two options:
If CHARSET(...) continues to refer to dt2, then it should have used_tables()=OUTER_REF_TABLE_BIT now. This is a bad approach to take as the optimizer doesn't currently support outer references inside derived tables.
Can we assume that CHARSET(dt2.dt2_c2) does not depend on dt2 and so it should have used_tables()=0?
bar, any objections to making CHARSET(table.column) have used_tables()=0 ? currently it seems to have arg[0]->used_tables() ?
Sergei Petrunia
added a comment - bar , any objections to making CHARSET(table.column) have used_tables()=0 ? currently it seems to have arg [0] ->used_tables() ?
The idea discussed during a meeting: move args[0]->charset_for_protocol() call to the preparation phase (fix_fields()) and remember the charset returned:
So we can safely set used_tables() to return 0 and avoid referring to arguments (args[0]) during the execution phase (call to val_str() is made during the execution).
Oleg Smirnov
added a comment - The idea discussed during a meeting: move args [0] ->charset_for_protocol() call to the preparation phase ( fix_fields() ) and remember the charset returned:
String *Item_func_charset::val_str(String *str)
{
DBUG_ASSERT(fixed());
uint dummy_errors;
CHARSET_INFO *cs= args[0]->charset_for_protocol();
null_value= 0;
str->copy(cs->cs_name.str, cs->cs_name.length,
&my_charset_latin1, collation.collation, &dummy_errors);
return str;
}
So we can safely set used_tables() to return 0 and avoid referring to arguments (args [0] ) during the execution phase (call to val_str() is made during the execution).
I have just one small suggestion: Please change the data type of the new member from:
CHARSET_INFO *m_cached_charset_info;
to
LEX_CSTRING m_cached_charset_info;
to avoid strlen() evaluation per row by evaluating it during fix_length_and_dec() instead.
Note, starting from 10.6 CHARSET_INFO was modified:
const char *csname;
was changed to:
LEX_CSTRING cs_name;
So starting from 10.6 strlen() won't be needed even in fix_length_and_dec().
Alexander Barkov
added a comment - Hi oleg.smirnov ,
the patch is OK to push.
I have just one small suggestion: Please change the data type of the new member from:
CHARSET_INFO *m_cached_charset_info;
to
LEX_CSTRING m_cached_charset_info;
to avoid strlen() evaluation per row by evaluating it during fix_length_and_dec() instead.
Note, starting from 10.6 CHARSET_INFO was modified:
const char *csname;
was changed to:
LEX_CSTRING cs_name;
So starting from 10.6 strlen() won't be needed even in fix_length_and_dec().
Note for the changelog: if a query used a derived table DT (a CTE or a
mergeable VIEW would work as well) and the WHERE clause compared columns
of DT with the value of CHARSET() or COERCIBILITY() function, the query
could produce wrong result or crash. The cause was incorrect processing
of these functions by derived condition pushdown optimization ( https://mariadb.com/kb/en/condition-pushdown-into-derived-table-optimization/)
Sergei Petrunia
added a comment - Note for the changelog: if a query used a derived table DT (a CTE or a
mergeable VIEW would work as well) and the WHERE clause compared columns
of DT with the value of CHARSET() or COERCIBILITY() function, the query
could produce wrong result or crash. The cause was incorrect processing
of these functions by derived condition pushdown optimization (
https://mariadb.com/kb/en/condition-pushdown-into-derived-table-optimization/ )
People
Oleg Smirnov
John Jove
Votes:
0Vote for this issue
Watchers:
7Start watching this issue
Dates
Created:
Updated:
Resolved:
Git Integration
Error rendering 'com.xiplink.jira.git.jira_git_plugin:git-issue-webpanel'. Please contact your Jira administrators.
{"report":{"fcp":1137.2000000476837,"ttfb":327.7000000476837,"pageVisibility":"visible","entityId":126908,"key":"jira.project.issue.view-issue","isInitial":true,"threshold":1000,"elementTimings":{},"userDeviceMemory":8,"userDeviceProcessors":64,"apdex":0.5,"journeyId":"7b377112-6497-48eb-8da5-7ea5d13f5d14","navigationType":0,"readyForUser":1246.7999999523163,"redirectCount":0,"resourceLoadedEnd":1090.0999999046326,"resourceLoadedStart":334.7000000476837,"resourceTiming":[{"duration":193.59999990463257,"initiatorType":"link","name":"https://jira.mariadb.org/s/2c21342762a6a02add1c328bed317ffd-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/css/_super/batch.css","startTime":334.7000000476837,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":334.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":528.2999999523163,"responseStart":0,"secureConnectionStart":0},{"duration":193.70000004768372,"initiatorType":"link","name":"https://jira.mariadb.org/s/7ebd35e77e471bc30ff0eba799ebc151-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/css/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":335,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":335,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":528.7000000476837,"responseStart":0,"secureConnectionStart":0},{"duration":238.79999995231628,"initiatorType":"script","name":"https://jira.mariadb.org/s/0917945aaa57108d00c5076fea35e069-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/js/_super/batch.js?locale=en","startTime":335.09999990463257,"connectEnd":335.09999990463257,"connectStart":335.09999990463257,"domainLookupEnd":335.09999990463257,"domainLookupStart":335.09999990463257,"fetchStart":335.09999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":335.09999990463257,"responseEnd":573.8999998569489,"responseStart":573.8999998569489,"secureConnectionStart":335.09999990463257},{"duration":330.59999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/s/2d8175ec2fa4c816e8023260bd8c1786-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/js/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":335.2999999523163,"connectEnd":335.2999999523163,"connectStart":335.2999999523163,"domainLookupEnd":335.2999999523163,"domainLookupStart":335.2999999523163,"fetchStart":335.2999999523163,"redirectEnd":0,"redirectStart":0,"requestStart":335.2999999523163,"responseEnd":665.8999998569489,"responseStart":665.8999998569489,"secureConnectionStart":335.2999999523163},{"duration":334,"initiatorType":"script","name":"https://jira.mariadb.org/s/a9324d6758d385eb45c462685ad88f1d-CDN/lu2cib/820016/12ta74/c92c0caa9a024ae85b0ebdbed7fb4bd7/_/download/contextbatch/js/atl.global,-_super/batch.js?locale=en","startTime":335.7000000476837,"connectEnd":335.7000000476837,"connectStart":335.7000000476837,"domainLookupEnd":335.7000000476837,"domainLookupStart":335.7000000476837,"fetchStart":335.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":335.7000000476837,"responseEnd":669.7000000476837,"responseStart":669.7000000476837,"secureConnectionStart":335.7000000476837},{"duration":334.2999999523163,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-en/jira.webresources:calendar-en.js","startTime":335.7999999523163,"connectEnd":335.7999999523163,"connectStart":335.7999999523163,"domainLookupEnd":335.7999999523163,"domainLookupStart":335.7999999523163,"fetchStart":335.7999999523163,"redirectEnd":0,"redirectStart":0,"requestStart":335.7999999523163,"responseEnd":670.0999999046326,"responseStart":670.0999999046326,"secureConnectionStart":335.7999999523163},{"duration":334.39999985694885,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-localisation-moment/jira.webresources:calendar-localisation-moment.js","startTime":336,"connectEnd":336,"connectStart":336,"domainLookupEnd":336,"domainLookupStart":336,"fetchStart":336,"redirectEnd":0,"redirectStart":0,"requestStart":336,"responseEnd":670.3999998569489,"responseStart":670.3999998569489,"secureConnectionStart":336},{"duration":403.7000000476837,"initiatorType":"link","name":"https://jira.mariadb.org/s/b04b06a02d1959df322d9cded3aeecc1-CDN/lu2cib/820016/12ta74/a2ff6aa845ffc9a1d22fe23d9ee791fc/_/download/contextbatch/css/jira.global.look-and-feel,-_super/batch.css","startTime":336.09999990463257,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":336.09999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":739.7999999523163,"responseStart":0,"secureConnectionStart":0},{"duration":334.59999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/rest/api/1.0/shortcuts/820016/47140b6e0a9bc2e4913da06536125810/shortcuts.js?context=issuenavigation&context=issueaction","startTime":336.2999999523163,"connectEnd":336.2999999523163,"connectStart":336.2999999523163,"domainLookupEnd":336.2999999523163,"domainLookupStart":336.2999999523163,"fetchStart":336.2999999523163,"redirectEnd":0,"redirectStart":0,"requestStart":336.2999999523163,"responseEnd":670.8999998569489,"responseStart":670.8999998569489,"secureConnectionStart":336.2999999523163},{"duration":403.60000014305115,"initiatorType":"link","name":"https://jira.mariadb.org/s/3ac36323ba5e4eb0af2aa7ac7211b4bb-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/css/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.css?jira.create.linked.issue=true","startTime":336.39999985694885,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":336.39999985694885,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":740,"responseStart":0,"secureConnectionStart":0},{"duration":334.90000009536743,"initiatorType":"script","name":"https://jira.mariadb.org/s/5d5e8fe91fbc506585e83ea3b62ccc4b-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/js/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.js?jira.create.linked.issue=true&locale=en","startTime":336.59999990463257,"connectEnd":336.59999990463257,"connectStart":336.59999990463257,"domainLookupEnd":336.59999990463257,"domainLookupStart":336.59999990463257,"fetchStart":336.59999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":336.59999990463257,"responseEnd":671.5,"responseStart":671.5,"secureConnectionStart":336.59999990463257},{"duration":582,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-js/jira.webresources:bigpipe-js.js","startTime":337.59999990463257,"connectEnd":337.59999990463257,"connectStart":337.59999990463257,"domainLookupEnd":337.59999990463257,"domainLookupStart":337.59999990463257,"fetchStart":337.59999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":337.59999990463257,"responseEnd":919.5999999046326,"responseStart":919.5999999046326,"secureConnectionStart":337.59999990463257},{"duration":745.8999998569489,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-init/jira.webresources:bigpipe-init.js","startTime":344.2000000476837,"connectEnd":344.2000000476837,"connectStart":344.2000000476837,"domainLookupEnd":344.2000000476837,"domainLookupStart":344.2000000476837,"fetchStart":344.2000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":344.2000000476837,"responseEnd":1090.0999999046326,"responseStart":1090.0999999046326,"secureConnectionStart":344.2000000476837},{"duration":149.60000014305115,"initiatorType":"xmlhttprequest","name":"https://jira.mariadb.org/rest/webResources/1.0/resources","startTime":770.3999998569489,"connectEnd":770.3999998569489,"connectStart":770.3999998569489,"domainLookupEnd":770.3999998569489,"domainLookupStart":770.3999998569489,"fetchStart":770.3999998569489,"redirectEnd":0,"redirectStart":0,"requestStart":770.3999998569489,"responseEnd":920,"responseStart":920,"secureConnectionStart":770.3999998569489},{"duration":395.40000009536743,"initiatorType":"link","name":"https://jira.mariadb.org/s/d5715adaadd168a9002b108b2b039b50-CDN/lu2cib/820016/12ta74/be4b45e9cec53099498fa61c8b7acba4/_/download/contextbatch/css/jira.project.sidebar,-_super,-project.issue.navigator,-jira.general,-jira.browse.project,-jira.view.issue,-jira.global,-atl.general,-com.atlassian.jira.projects.sidebar.init/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":1044.8999998569489,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":1044.8999998569489,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1440.2999999523163,"responseStart":0,"secureConnectionStart":0}],"fetchStart":0,"domainLookupStart":0,"domainLookupEnd":0,"connectStart":0,"connectEnd":0,"requestStart":113,"responseStart":328,"responseEnd":344,"domLoading":332,"domInteractive":1354,"domContentLoadedEventStart":1354,"domContentLoadedEventEnd":1435,"domComplete":1793,"loadEventStart":1794,"loadEventEnd":1794,"userAgent":"Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)","marks":[{"name":"bigPipe.sidebar-id.start","time":1291.5999999046326},{"name":"bigPipe.sidebar-id.end","time":1292.3999998569489},{"name":"bigPipe.activity-panel-pipe-id.start","time":1292.5},{"name":"bigPipe.activity-panel-pipe-id.end","time":1301.2000000476837},{"name":"activityTabFullyLoaded","time":1461}],"measures":[],"correlationId":"24be65739ebc57","effectiveType":"4g","downlink":10,"rtt":0,"serverDuration":144,"dbReadsTimeInMs":12,"dbConnsTimeInMs":21,"applicationHash":"9d11dbea5f4be3d4cc21f03a88dd11d8c8687422","experiments":[]}}
When removing the key build in column c1 in table t1, the error is gone.