home *** CD-ROM | disk | FTP | other *** search
- Rem Copyright (c) 1989 by Oracle Corporation
- Rem NAME
- Rem locktree.sql
- Rem FUNCTION - Print out the lock wait-for graph in tree structured fashion.
- Rem This is useful for diagnosing systems that are hung on locks.
- Rem NOTES
- Rem MODIFIED
- Rem Loaiza 11/01/89 - Creation
- Rem
-
- /* Print out the lock wait-for graph in a tree structured fashion.
- *
- * This script prints the processes in the system that are waiting for
- * locks, and the locks that they are waiting for. The printout is tree
- * structured. If a processid is printed immediately below and to the right
- * of another process, then it is waiting for that process. The process ids
- * printed at the left hand side of the page are the ones that everyone is
- * waiting for.
- *
- * For example, in the following printout process 9 is waiting for
- * process 8, 7 is waiting for 9, and 10 is waiting for 9.
- *
- * PROCESS WAITING TYPE MODE REQUESTED MODE HELD LOCK ID1 LOCK ID2
- * ----------------- ---- ----------------- ----------------- -------- --------
- * 8 NONE None None 0 0
- * 9 TX Share (S) Exclusive (X) 65547 16
- * 7 RW Exclusive (X) S/Row-X (SSX) 33554440 2
- * 10 RW Exclusive (X) S/Row-X (SSX) 33554440 2
- *
- * The lock information to the right of the process id describes the lock
- * that the process is waiting for (not the lock it is holding).
- *
- * You must have the blocking.sql script loaded for this script to work.
- *
- * Note that this is a script and not a set of view definitions because
- * connect-by is used in the implementation and therefore a temporary table
- * is created and dropped since you cannot do a join in a connect-by.
- *
- * This script has two small disadvantages. One, a table is created when
- * this script is run. To create a table a number of locks must be
- * acquired. This might cause the process running the script to get caught
- * in the lock problem it is trying to diagnose. Two, if a process waits on
- * a lock held by more than one process (share lock) then the wait-for graph
- * is no longer a tree and the conenct-by will show the process (and any
- * processes waiting on it) several times.
- */
-
-
- /* Select all pids waiting for a lock, the lock they are waiting on, and the
- * pid of the process that holds the lock.
- * UNION
- * The pids of all processes holding locks that someone is waiting on that
- * are not themselves waiting for locks. These are included so that the roots
- * of the wait for graph (the processes holding things up) will be displayed.
- *
- * pid - pid of requesting process
- * type - type of lock being requested
- * id1 - id1 of lock being requested (value is lock type specific)
- * id2 - id2 of lock being requested (value is lock type specific)
- * req - mode lock is being requested in
- * hpid - pid of process holding the lock
- * hmod - mode the lock is held in
- */
- create table LOCK_HOLDERS
- (
- pid number(18),
- type char(4),
- id1 char(10),
- id2 char(10),
- req number,
- hpid number,
- hmod number
- );
-
- insert into lock_holders
- select r.pid, r.type, r.id1, r.id2, r.request req, h.pid hpid, h.lmode hmod
- from all_locks_view h, all_locks_view r
- where r.request > 1 /* R is waiting for (requesting) the lock */
- and h.lmode > 1 /* H is holding the lock */
- and h.type = r.type /* H is holding the lock R is requesting */
- and h.id1 = r.id1
- and h.id2 = r.id2
- union
- select unique
- h.pid, 'NONE', '0', '0', 0, -1, 0
- from v$process p, all_locks_view h, all_locks_view r
- where r.request > 1 /* R is waiting for (requesting) the lock */
- and h.lmode > 1 /* h is holding the lock */
- and h.type = r.type /* H is holding the lock R is requesting */
- and h.id1 = r.id1
- and h.id2 = r.id2
- and h.pid = p.pid /* H is process P */
- and p.lockwait is null /* P is not waiting for a lock */
- and p.latchwait is null;
-
- set charwidth 17;
-
- /* Print out the result in a tree structured fashion */
-
- select lpad(' ',3*(level-1)) || pid "PROCESS WAITING",
- type type,
- decode( req,
- 0, 'None', /* Mon Lock equivalent */
- 1, 'Null Mode', /* N */
- 2, 'Row-S (SS)', /* L */
- 3, 'Row-X (SX)', /* R */
- 4, 'Share (S)', /* S */
- 5, 'S/Row-X (SSX)', /* C */
- 6, 'Exclusive (X)', /* X */
- req) "MODE REQUESTED",
- decode( hmod,
- 0, 'None', /* Mon Lock equivalent */
- 1, 'Null Mode', /* N */
- 2, 'Row-S (SS)', /* L */
- 3, 'Row-X (SX)', /* R */
- 4, 'Share (S)', /* S */
- 5, 'S/Row-X (SSX)', /* C */
- 6, 'Exclusive (X)', /* X */
- hmod) "MODE HELD",
- id1 "LOCK ID1", id2 "LOCK ID2"
- from lock_holders
- connect by prior pid = hpid
- start with hpid = -1;
-
- drop table lock_holders;
-