summaryrefslogtreecommitdiffstats
path: root/include/linux/lockdep.h
diff options
context:
space:
mode:
authorByungchul Park <byungchul.park@lge.com>2017-08-07 03:12:54 -0400
committerIngo Molnar <mingo@kernel.org>2017-08-10 06:29:08 -0400
commit28a903f63ec0811ead70ad0f8665e838d207a25e (patch)
treebc060a5056284d7e030c22158a6708f97c25ad51 /include/linux/lockdep.h
parent23f873d8f9526ed7e49a1a02a45f8afb9ae5fb84 (diff)
locking/lockdep: Handle non(or multi)-acquisition of a crosslock
No acquisition might be in progress on commit of a crosslock. Completion operations enabling crossrelease are the case like: CONTEXT X CONTEXT Y --------- --------- trigger completion context complete AX commit AX wait_for_complete AX acquire AX wait where AX is a crosslock. When no acquisition is in progress, we should not perform commit because the lock does not exist, which might cause incorrect memory access. So we have to track the number of acquisitions of a crosslock to handle it. Moreover, in case that more than one acquisition of a crosslock are overlapped like: CONTEXT W CONTEXT X CONTEXT Y CONTEXT Z --------- --------- --------- --------- acquire AX (gen_id: 1) acquire A acquire AX (gen_id: 10) acquire B commit AX acquire C commit AX where A, B and C are typical locks and AX is a crosslock. Current crossrelease code performs commits in Y and Z with gen_id = 10. However, we can use gen_id = 1 to do it, since not only 'acquire AX in X' but 'acquire AX in W' also depends on each acquisition in Y and Z until their commits. So make it use gen_id = 1 instead of 10 on their commits, which adds an additional dependency 'AX -> A' in the example above. Signed-off-by: Byungchul Park <byungchul.park@lge.com> Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: akpm@linux-foundation.org Cc: boqun.feng@gmail.com Cc: kernel-team@lge.com Cc: kirill@shutemov.name Cc: npiggin@gmail.com Cc: walken@google.com Cc: willy@infradead.org Link: http://lkml.kernel.org/r/1502089981-21272-8-git-send-email-byungchul.park@lge.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'include/linux/lockdep.h')
-rw-r--r--include/linux/lockdep.h22
1 files changed, 21 insertions, 1 deletions
diff --git a/include/linux/lockdep.h b/include/linux/lockdep.h
index c75eedd55af5..651cc61af041 100644
--- a/include/linux/lockdep.h
+++ b/include/linux/lockdep.h
@@ -325,6 +325,19 @@ struct hist_lock {
325 */ 325 */
326struct cross_lock { 326struct cross_lock {
327 /* 327 /*
328 * When more than one acquisition of crosslocks are overlapped,
329 * we have to perform commit for them based on cross_gen_id of
330 * the first acquisition, which allows us to add more true
331 * dependencies.
332 *
333 * Moreover, when no acquisition of a crosslock is in progress,
334 * we should not perform commit because the lock might not exist
335 * any more, which might cause incorrect memory access. So we
336 * have to track the number of acquisitions of a crosslock.
337 */
338 int nr_acquire;
339
340 /*
328 * Seperate hlock instance. This will be used at commit step. 341 * Seperate hlock instance. This will be used at commit step.
329 * 342 *
330 * TODO: Use a smaller data structure containing only necessary 343 * TODO: Use a smaller data structure containing only necessary
@@ -547,9 +560,16 @@ extern void lockdep_init_map_crosslock(struct lockdep_map *lock,
547 int subclass); 560 int subclass);
548extern void lock_commit_crosslock(struct lockdep_map *lock); 561extern void lock_commit_crosslock(struct lockdep_map *lock);
549 562
563/*
564 * What we essencially have to initialize is 'nr_acquire'. Other members
565 * will be initialized in add_xlock().
566 */
567#define STATIC_CROSS_LOCK_INIT() \
568 { .nr_acquire = 0,}
569
550#define STATIC_CROSS_LOCKDEP_MAP_INIT(_name, _key) \ 570#define STATIC_CROSS_LOCKDEP_MAP_INIT(_name, _key) \
551 { .map.name = (_name), .map.key = (void *)(_key), \ 571 { .map.name = (_name), .map.key = (void *)(_key), \
552 .map.cross = 1, } 572 .map.cross = 1, .xlock = STATIC_CROSS_LOCK_INIT(), }
553 573
554/* 574/*
555 * To initialize a lockdep_map statically use this macro. 575 * To initialize a lockdep_map statically use this macro.