[MDEV-17073] INSERT…ON DUPLICATE KEY UPDATE became more deadlock-prone Created: 2018-08-27 Updated: 2020-08-25 Resolved: 2018-11-02 |
|
| Status: | Closed |
| Project: | MariaDB Server |
| Component/s: | Storage Engine - InnoDB |
| Affects Version/s: | 10.2.2, 10.3.0 |
| Fix Version/s: | 10.3.11, 10.2.19 |
| Type: | Bug | Priority: | Major |
| Reporter: | Marko Mäkelä | Assignee: | Marko Mäkelä |
| Resolution: | Fixed | Votes: | 3 |
| Labels: | upstream | ||
| Issue Links: |
|
||||||||||||||||||||||||||||||||||||||||
| Description |
|
MySQL 5.7.4 changed the behaviour of INSERT…ON DUPLICATE KEY UPDATE in the InnoDB storage engine. Upon encountering a duplicate key, it would no longer directly fall back to INSERT, but instead it would proceed to acquire an exclusive lock on every index record for the row on which the UPDATE failed. The extra locking was motivated by a public bug report: MySQL Bug#50413 insert on duplicate key update sometimes writes binlog position incorrectly (Oracle internal BUG#11758237). The fix was followed up by a couple of regression fixes. For one user, reverting these changes significantly reduces the deadlock rate of INSERT…ON DUPLICATE KEY UPDATE. There also is a related MySQL Bug #52020 InnoDB can still deadlock on just INSERT...ON DUPLICATE KEY. One of the factors was that when the pluggable storage engine interface was created in MySQL 5.1, the function innobase_query_is_update() was removed without replacement, and MySQL Bug #7975 (which lacked a test case) was reintroduced. In a comment in MySQL Bug #52020 I anticipated that the deadlocks would be caused in a scenario where the INSERT phase fails, then some other transaction locks some of the index records, causing the ON DUPLICATE KEY UPDATE phase to wait for those locks or to deadlock. Acquiring the locks for all index records already in the INSERT phase would make the UPDATE phase wait-free, but it could cause more conflicts with other accesses, as the hold time of the locks is extended. valerii posted some insightful comments on Bug #52020. I hope he can construct a test case that demonstrates the increased deadlock rate, so that we can see what can be improved here. Sven Sandberg suggested in MySQL Bug #50413 that the INSERT phase should have acquired a gap lock, so that conflicting INSERT with that key would be prevented. I assume that he meant the PRIMARY key, because his example involves two unique keys: PRIMARY KEY(a), UNIQUE KEY(b). He also filed MySQL Bug #58637 Mark INSERT...ON DUPLICATE KEY UPDATE unsafe when there is more than one key. Apparently the nondeterminism that the extra locking is trying to prevent is caused by the ambiguity of the ON DUPLICATE KEY syntax. It does not specify the key! An unambiguous syntax would be something like:
Statement-based replication is obviously affected by this ambiguity. Note: Comments in MySQL Bug #50413 suggest that innodb_autoinc_lock_mode settings 0 and 1 are equivalent in this respect. I’d also like to know whether this parameter is at all relevant outside statement-based replication (that is, when innodb_autoinc_lock_mode=2 could be safe to use). With the setting 2, InnoDB does not acquire any auto-increment lock within the transaction. With the settings 0 or 1, InnoDB will hold a lock until the end of the current statement. This would suggest that the setting only matters in statement-based replication. |
| Comments |
| Comment by Seppo Jaakola [ 2018-08-27 ] | ||||||||||||||||
|
Galera replication exercises optimistic concurrency control, whatever happens during the transaction processing, in master node, does not really matter. At commit time, Galera populates a replication write set which will contain binlog events for the transaction and key information for modified rows (primary keys, unique keys and foreign keys). If insert succeeds, there should be write rows events, and if insert execution deviated for updating, there should be update rows events in the binlog events set. In the slave node side, the write set is applied directly by using primary keys in the respective binlog events, so this is rather straightforward operation, and does not involve excessive locking. However, there will be some harm for replication performance, in multi-master topologies, if transaction locks more rows than what will be needed during the write set applying phase in slave node. In such "asymmetric locking situation", the INSERT...ON DUPLICATE... execution, which has advanced in replication phase, may still end up as victim for earlier replicated transaction, and it has to rollback. If the conflict happened over such locks, which are not used in applying phase, Galera will still abort the transaction and immediately replay. This rollback-replay cycle, in master node, will slow down the master node somewhat, especially if such non related conflicts are frequent. | ||||||||||||||||
| Comment by Andrei Elkin [ 2018-09-11 ] | ||||||||||||||||
|
marko, thanks for a well-conducted analysis and references. Indeed the unsafety of IODKU is caused | ||||||||||||||||
| Comment by Marko Mäkelä [ 2018-09-11 ] | ||||||||||||||||
|
Andrei, I hope you can run some tests and suggest what should be done. I see two possible outcomes, but I may be wrong:
If the outcome is the first one, then I would appreciate your help in writing a predicate function. | ||||||||||||||||
| Comment by Andrei Elkin [ 2018-09-11 ] | ||||||||||||||||
|
marko There always has existed a method to defeat non-determinism without involving engine. As which key to chose to detect a conflict | ||||||||||||||||
| Comment by Marko Mäkelä [ 2018-09-18 ] | ||||||||||||||||
|
I see locking as a way to make things serialized or more deterministic in a distributed system. We discussed three main sources of nondeterminism between a master and a slave:
In InnoDB, the PRIMARY KEY or the clustered index is always modified first. On CREATE TABLE or after rebuilding a table with ALGORITHM=COPY, InnoDB would create the dict_table_t::indexes in the order of TABLE::key_info[], that is, UNIQUE KEY before non-unique ones. It makes sense to process unique keys first, to avoid modifying non-unique indexes in case a duplicate key error occurs. If ALTER TABLE…ADD UNIQUE KEY or CREATE UNIQUE INDEX was executed with ALGORITHM=INPLACE, then InnoDB would add the index last (while it would be among the first ones in the .frm file. If the master and slaves use different settings for old_alter_table, or if some system was initialized from mysqldump while others are running on files where an unique index was created with ALGORITHM=INPLACE, then InnoDB would be locking and processing the index records in different order. It seems to me that when statement-based replication is used, we must keep the current level of locking. For row-based replication, Elkin suggested that the replication event should indicate the set of indexes where locks were acquired in the INSERT phase before the ON DUPLICATE KEY kicked in. The slave would then ensure that the equivalent locks will be acquired. In this way, the INSERT phase could avoid locking the records in all indexes, thus reducing the potential for lock conflicts. Reducing the number of locks acquired will not only reduce the potential for deadlocks, but also the potential for lock waits and lock wait timeouts. | ||||||||||||||||
| Comment by Geoff Montee (Inactive) [ 2018-10-22 ] | ||||||||||||||||
I created | ||||||||||||||||
| Comment by Marko Mäkelä [ 2018-11-02 ] | ||||||||||||||||
|
As a simple fix, I would keep the extra locking only when binlog is enabled and statement-based replication is in use. In this way, we will get deterministic locking on master and slave, while not paying penalty for standalone operation or row-based replication:
| ||||||||||||||||
| Comment by Marko Mäkelä [ 2018-11-02 ] | ||||||||||||||||
|
I pushed a fix to 10.2. With the following patch, it trips a check at the upper level:
Maybe a better fix would be to revert the upstream change entirely and improve the check that was originally added in MySQL 5.1.20 in ha_innobase::table_flags() and in 5.1.21 moved to ha_innobase::external_lock(). | ||||||||||||||||
| Comment by Marko Mäkelä [ 2019-04-25 ] | ||||||||||||||||
|
It turns out that upstream reverted the problematic fix and implemented a cleaner fix in MySQL 5.7.26. | ||||||||||||||||
| Comment by Marko Mäkelä [ 2019-08-12 ] | ||||||||||||||||
|
In MariaDB Server 10.2 and later, thanks to |