* [PATCH] txn: don't unref stmt tuples before rollback triggers are run
@ 2019-07-18 15:45 Vladimir Davydov
2019-07-19 17:40 ` Vladimir Davydov
2019-07-24 12:30 ` [tarantool-patches] " Георгий Кириченко
0 siblings, 2 replies; 6+ messages in thread
From: Vladimir Davydov @ 2019-07-18 15:45 UTC (permalink / raw)
To: kostja; +Cc: tarantool-patches
Rollback triggers expect to see the database state after reverting all
changes done by the transaction. At the same time they may use tuples
referenced by reverted statements.
As a result, using txn_rollback_to_svp(), which both reverts changes and
releases tuples, on yield in a DDL transaction may result in a crash.
The crash can only be reproduced using SQL, because Lua implicitly
references all tuples anyway.
Thanks to @Gerold103 for reporting this issue and carrying out
preliminary investigation.
Closes #4365
---
https://github.com/tarantool/tarantool/issues/4365
https://github.com/tarantool/tarantool/commits/dv/gh-4365-fix-tx-ddl-crash-on-yield
src/box/txn.c | 22 +++++++++++++++++++++-
test/box/transaction.result | 9 +++++++++
test/box/transaction.test.lua | 6 ++++++
3 files changed, 36 insertions(+), 1 deletion(-)
diff --git a/src/box/txn.c b/src/box/txn.c
index 73eaee73..6d844af2 100644
--- a/src/box/txn.c
+++ b/src/box/txn.c
@@ -761,10 +761,30 @@ txn_on_yield(struct trigger *trigger, void *event)
(void) event;
struct txn *txn = in_txn();
assert(txn != NULL && !txn->can_yield);
- txn_rollback_to_svp(txn, NULL);
+ /*
+ * Rollback triggers expect to see the database state
+ * after reverting all changes done by the transaction.
+ * At the same time they may use tuples referenced by
+ * reverted statements. So we rollback statements before
+ * running the triggers, but postpone tuple release until
+ * after the triggers have been invoked.
+ */
+ struct txn_stmt *stmt;
+ stailq_foreach_entry(stmt, &txn->stmts, next) {
+ if (stmt->space != NULL)
+ engine_rollback_statement(txn->engine, txn, stmt);
+ stmt->space = NULL;
+ stmt->row = NULL;
+ }
if (txn->has_triggers) {
txn_run_triggers(txn, &txn->on_rollback);
txn->has_triggers = false;
}
+ stailq_foreach_entry(stmt, &txn->stmts, next)
+ txn_stmt_unref_tuples(stmt);
+ stailq_create(&txn->stmts);
+ txn->n_new_rows = 0;
+ txn->n_local_rows = 0;
+ txn->n_applier_rows = 0;
txn->is_aborted_by_yield = true;
}
diff --git a/test/box/transaction.result b/test/box/transaction.result
index 9a48f2f7..771ad96e 100644
--- a/test/box/transaction.result
+++ b/test/box/transaction.result
@@ -778,3 +778,12 @@ box.begin() drop() box.rollback()
box.begin() drop() box.commit()
---
...
+--
+-- gh-4365: DDL revered by yield triggers crash.
+--
+box.begin() box.execute([[CREATE TABLE test(id INTEGER PRIMARY KEY AUTOINCREMENT)]]) fiber.yield()
+---
+...
+box.rollback()
+---
+...
diff --git a/test/box/transaction.test.lua b/test/box/transaction.test.lua
index 0792cc3c..d8a6e60a 100644
--- a/test/box/transaction.test.lua
+++ b/test/box/transaction.test.lua
@@ -415,3 +415,9 @@ box.begin() create() box.rollback()
box.begin() create() box.commit()
box.begin() drop() box.rollback()
box.begin() drop() box.commit()
+
+--
+-- gh-4365: DDL revered by yield triggers crash.
+--
+box.begin() box.execute([[CREATE TABLE test(id INTEGER PRIMARY KEY AUTOINCREMENT)]]) fiber.yield()
+box.rollback()
--
2.11.0
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH] txn: don't unref stmt tuples before rollback triggers are run
2019-07-18 15:45 [PATCH] txn: don't unref stmt tuples before rollback triggers are run Vladimir Davydov
@ 2019-07-19 17:40 ` Vladimir Davydov
2019-07-19 19:28 ` [tarantool-patches] " Vladislav Shpilevoy
2019-07-24 12:30 ` [tarantool-patches] " Георгий Кириченко
1 sibling, 1 reply; 6+ messages in thread
From: Vladimir Davydov @ 2019-07-19 17:40 UTC (permalink / raw)
To: kostja; +Cc: tarantool-patches
Ignore this patch please. Will send a new version soon.
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [tarantool-patches] Re: [PATCH] txn: don't unref stmt tuples before rollback triggers are run
2019-07-19 17:40 ` Vladimir Davydov
@ 2019-07-19 19:28 ` Vladislav Shpilevoy
2019-07-19 19:36 ` Vladislav Shpilevoy
0 siblings, 1 reply; 6+ messages in thread
From: Vladislav Shpilevoy @ 2019-07-19 19:28 UTC (permalink / raw)
To: tarantool-patches, Vladimir Davydov, kostja
Why? What is wrong?
On 19/07/2019 19:40, Vladimir Davydov wrote:
> Ignore this patch please. Will send a new version soon.
>
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [tarantool-patches] Re: [PATCH] txn: don't unref stmt tuples before rollback triggers are run
2019-07-19 19:28 ` [tarantool-patches] " Vladislav Shpilevoy
@ 2019-07-19 19:36 ` Vladislav Shpilevoy
0 siblings, 0 replies; 6+ messages in thread
From: Vladislav Shpilevoy @ 2019-07-19 19:36 UTC (permalink / raw)
To: tarantool-patches, Vladimir Davydov, kostja
Never mind, I see now.
On 19/07/2019 21:28, Vladislav Shpilevoy wrote:
> Why? What is wrong?
>
> On 19/07/2019 19:40, Vladimir Davydov wrote:
>> Ignore this patch please. Will send a new version soon.
>>
>
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [tarantool-patches] [PATCH] txn: don't unref stmt tuples before rollback triggers are run
2019-07-18 15:45 [PATCH] txn: don't unref stmt tuples before rollback triggers are run Vladimir Davydov
2019-07-19 17:40 ` Vladimir Davydov
@ 2019-07-24 12:30 ` Георгий Кириченко
2019-07-24 12:55 ` Vladimir Davydov
1 sibling, 1 reply; 6+ messages in thread
From: Георгий Кириченко @ 2019-07-24 12:30 UTC (permalink / raw)
To: tarantool-patches; +Cc: Vladimir Davydov, kostja
[-- Attachment #1: Type: text/plain, Size: 3424 bytes --]
Could we consider the possibility to assign triggers on a statement itself?
In this case you shouldn't implement all that machinery with savepoints.
On Thursday, July 18, 2019 6:45:56 PM MSK Vladimir Davydov wrote:
> Rollback triggers expect to see the database state after reverting all
> changes done by the transaction. At the same time they may use tuples
> referenced by reverted statements.
>
> As a result, using txn_rollback_to_svp(), which both reverts changes and
> releases tuples, on yield in a DDL transaction may result in a crash.
> The crash can only be reproduced using SQL, because Lua implicitly
> references all tuples anyway.
>
> Thanks to @Gerold103 for reporting this issue and carrying out
> preliminary investigation.
>
> Closes #4365
> ---
> https://github.com/tarantool/tarantool/issues/4365
> https://github.com/tarantool/tarantool/commits/dv/gh-4365-fix-tx-ddl-crash-o
> n-yield
>
> src/box/txn.c | 22 +++++++++++++++++++++-
> test/box/transaction.result | 9 +++++++++
> test/box/transaction.test.lua | 6 ++++++
> 3 files changed, 36 insertions(+), 1 deletion(-)
>
> diff --git a/src/box/txn.c b/src/box/txn.c
> index 73eaee73..6d844af2 100644
> --- a/src/box/txn.c
> +++ b/src/box/txn.c
> @@ -761,10 +761,30 @@ txn_on_yield(struct trigger *trigger, void *event)
> (void) event;
> struct txn *txn = in_txn();
> assert(txn != NULL && !txn->can_yield);
> - txn_rollback_to_svp(txn, NULL);
> + /*
> + * Rollback triggers expect to see the database state
> + * after reverting all changes done by the transaction.
> + * At the same time they may use tuples referenced by
> + * reverted statements. So we rollback statements before
> + * running the triggers, but postpone tuple release until
> + * after the triggers have been invoked.
> + */
> + struct txn_stmt *stmt;
> + stailq_foreach_entry(stmt, &txn->stmts, next) {
> + if (stmt->space != NULL)
> + engine_rollback_statement(txn->engine, txn, stmt);
> + stmt->space = NULL;
> + stmt->row = NULL;
> + }
> if (txn->has_triggers) {
> txn_run_triggers(txn, &txn->on_rollback);
> txn->has_triggers = false;
> }
> + stailq_foreach_entry(stmt, &txn->stmts, next)
> + txn_stmt_unref_tuples(stmt);
> + stailq_create(&txn->stmts);
> + txn->n_new_rows = 0;
> + txn->n_local_rows = 0;
> + txn->n_applier_rows = 0;
> txn->is_aborted_by_yield = true;
> }
> diff --git a/test/box/transaction.result b/test/box/transaction.result
> index 9a48f2f7..771ad96e 100644
> --- a/test/box/transaction.result
> +++ b/test/box/transaction.result
> @@ -778,3 +778,12 @@ box.begin() drop() box.rollback()
> box.begin() drop() box.commit()
> ---
> ...
> +--
> +-- gh-4365: DDL revered by yield triggers crash.
> +--
> +box.begin() box.execute([[CREATE TABLE test(id INTEGER PRIMARY KEY
> AUTOINCREMENT)]]) fiber.yield() +---
> +...
> +box.rollback()
> +---
> +...
> diff --git a/test/box/transaction.test.lua b/test/box/transaction.test.lua
> index 0792cc3c..d8a6e60a 100644
> --- a/test/box/transaction.test.lua
> +++ b/test/box/transaction.test.lua
> @@ -415,3 +415,9 @@ box.begin() create() box.rollback()
> box.begin() create() box.commit()
> box.begin() drop() box.rollback()
> box.begin() drop() box.commit()
> +
> +--
> +-- gh-4365: DDL revered by yield triggers crash.
> +--
> +box.begin() box.execute([[CREATE TABLE test(id INTEGER PRIMARY KEY
> AUTOINCREMENT)]]) fiber.yield() +box.rollback()
[-- Attachment #2: This is a digitally signed message part. --]
[-- Type: application/pgp-signature, Size: 488 bytes --]
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [tarantool-patches] [PATCH] txn: don't unref stmt tuples before rollback triggers are run
2019-07-24 12:30 ` [tarantool-patches] " Георгий Кириченко
@ 2019-07-24 12:55 ` Vladimir Davydov
0 siblings, 0 replies; 6+ messages in thread
From: Vladimir Davydov @ 2019-07-24 12:55 UTC (permalink / raw)
To: Георгий
Кириченко
Cc: tarantool-patches, kostja
On Wed, Jul 24, 2019 at 03:30:47PM +0300, Георгий Кириченко wrote:
> Could we consider the possibility to assign triggers on a statement itself?
> In this case you shouldn't implement all that machinery with savepoints.
FYI I scrapped this patch, as I wrote in reply to this email:
https://www.freelists.org/post/tarantool-patches/PATCH-txn-dont-unref-stmt-tuples-before-rollback-triggers-are-run,1
Instead I made savepoints support commit/rollback triggers:
https://www.freelists.org/post/tarantool-patches/PATCH-04-Support-savepoints-in-DDL-transactions
Regarding assigning commit/rollback triggers to individual statements.
Although this would indeed work, I prefer not to do that, because it's
somewhat expensive from the memory usage pov. Besides splitting
commit/rollback trigger lists like that would complicate the code
involving the triggers (running, setting/clearing triggers from Lua).
^ permalink raw reply [flat|nested] 6+ messages in thread
end of thread, other threads:[~2019-07-24 12:55 UTC | newest]
Thread overview: 6+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2019-07-18 15:45 [PATCH] txn: don't unref stmt tuples before rollback triggers are run Vladimir Davydov
2019-07-19 17:40 ` Vladimir Davydov
2019-07-19 19:28 ` [tarantool-patches] " Vladislav Shpilevoy
2019-07-19 19:36 ` Vladislav Shpilevoy
2019-07-24 12:30 ` [tarantool-patches] " Георгий Кириченко
2019-07-24 12:55 ` Vladimir Davydov
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox