Using DTrace to understand why PL/SQL packages can be faster than stored procedures (kgiPinObject)


19.09.2016
by Kamil Stawiarski

I know – everyone knows, that PL/SQL packages are faster than stored procedures. If you’ll ask anyone at the training or in your dev team "what is better" – you’ll (almost) always hear: PL/SQL packages. But why exactly?

The documentation says:

The first time you invoke a package subprogram, Oracle Database loads the whole package into memory. Subsequent invocations of other subprograms in same the package require no disk I/O.

Source: http://docs.oracle.com/database/121/LNPLS/packages.htm#LNPLS00904

OK, it’s understandable and feels natural. But let’s try to think about it for a while.
I’ve created one procedure, one function and a package, containing those two. As simple as it can be:

1SQL> create or replace function by2(x number) return number as
2  2  begin
3  3    return x*2;
4  4  end;
5  5  /
6 
7Function created.
8 
9SQL> create or replace procedure proc_by2 as
10  2   x number;
11  3  begin
12  4    for i in 1..1000000 loop
13  5      x:=by2(i);
14  6    end loop;
15  7  end;
16  8  /
17 
18Procedure created.
19 
20SQL> create or replace package pkg_test_calls as
21  2    function by2(x number) return number;
22  3    procedure by2_proc;
23  4  end;
24  5  /
25 
26Package created.
27 
28SQL> ed
29Wrote file afiedt.buf
30 
31  1  create or replace package body pkg_test_calls as
32  2    function by2(x number) return number is
33  3    begin
34  4      return x*2;
35  5    end by2;
36  6    procedure by2_proc is
37  7      vx number;
38  8    begin
39  9       for i in 1..1000000 loop
40 10         vx:=by2(i);
41 11       end loop;
42 12    end;
43 13* end;
44 14  /
45 
46Package body created.

I think that for this simple example, using "set timing on" will be sufficient enough.

1SQL> set timing on
2SQL> exec proc_by2
3 
4PL/SQL procedure successfully completed.
5 
6Elapsed: 00:00:00.25
7SQL> exec pkg_test_calls.by2_proc
8 
9PL/SQL procedure successfully completed.
10 
11Elapsed: 00:00:00.16
12SQL> exec proc_by2
13 
14PL/SQL procedure successfully completed.
15 
16Elapsed: 00:00:00.23
17SQL> exec pkg_test_calls.by2_proc
18 
19PL/SQL procedure successfully completed.
20 
21Elapsed: 00:00:00.16
22SQL>

So why there is a difference here? I don’t see any place here for disk I/O. Both procedures were compiled and loaded into memory, so we can’t speak about subsequent loads.

To understand what has happened I’ve created a very simple DTrace script, which will count the number of function calls, made by server process, associated with my session. I’ve used DTrace ability to trace functions, based on PID provider.

This technique is called "User-space Probing" and can be done also on Linux, using dynamic tracing tools like Systemtap https://sourceware.org/systemtap/SystemTap_Beginners_Guide/userspace-probing.html

1root@skiper:~# cat function_calls.d
2#!/usr/sbin/dtrace -s
3pid$1:::entry
4/pid==$1/
5{
6        @fcount[probemod,probefunc]=count();
7}

Let’s check the tail output for test with package and separate PL/SQL stored function/procedure.

1root@skiper:~# tail -20 fcalls_proc_fun.out
2  oracle                                              lnxmin                                                      1000000
3  oracle                                              lnxmin_int                                                  1000000
4  oracle                                              lnxmul                                                      1000000
5  oracle                                              pfrinstr_BCTR                                               1000000
6  oracle                                              pfrinstr_CVTIN                                              1000000
7  oracle                                              pfrinstr_MULN                                               1000000
8  oracle                                              sslnxmul                                                    1000000
9  oracle                                              kgiPinObject                                                1000002
10  oracle                                              pevm_ENTER                                                  1000002
11  oracle                                              pevm_RET                                                    1000002
12  oracle                                              pfrinstr_RET                                                1000002
13  oracle                                              pfrust                                                      1000002
14  oracle                                              pfrxca                                                      1000002
15  oracle                                              pevm_CHK_INST                                               1000003
16  oracle                                              pfrinstr_XCAL                                               1000003
17  oracle                                              pfrinstr_XCAL_SCAL                                          1000003
18  oracle                                              kglPinSetContext                                            1000004
19  oracle                                              kglpnp                                                      1000006
20  oracle                                              kss_get_owner                                               1000012
21  oracle                                              kss_get_type                                                1000028
22root@skiper:~# tail -20 fcalls_pkg.out
23  oracle                                              pdz2M20_Audit_In_No_PDZ0_DEBUG                                  634
24  oracle                                              dbgtrPrepareWrite                                               635
25  oracle                                              pdz9M08_Type_Kind                                               689
26  libc.so.1                                           setjmp                                                          717
27  oracle                                              kghuwrlength                                                    755
28  oracle                                              pdzgM07_Get                                                     774
29  oracle                                              pdz7M12_Opcode                                                  792
30  libc.so.1                                           memset                                                         1200
31  oracle                                              lnxmul                                                      1000000
32  oracle                                              pfrinstr_BCTR                                               1000000
33  oracle                                              pfrinstr_CALL                                               1000000
34  oracle                                              pfrinstr_CVTIN                                              1000000
35  oracle                                              pfrinstr_MULN                                               1000000
36  oracle                                              sslnxmul                                                    1000000
37  oracle                                              lnxmin                                                      1000001
38  oracle                                              lnxmin_int                                                  1000001
39  oracle                                              pevm_ENTER                                                  1000002
40  oracle                                              pevm_RET                                                    1000002
41  oracle                                              pfrinstr_RET                                                1000002
42  oracle                                              pfrust                                                      1000002

We don’t need a very detail investigation to notice the difference between those two executions.
I decided to check the execution times of 4 particular functions:

  • plsql_run
  • kglpnp
  • kglPinSetContext
  • kgiPinObject

The first one is responsible for the whole PL/SQL initialization and execution, so its time should be similar to what we can see in SQL*Plus timing. The next two should have something to do with maintaining the PL/SQL objects in library cache – "oradebug doc component" shows, that "kgl" components are responsible for LIBRARY CACHE:

1ora-600:~ inter$ grep -i kgl ORA-600/scripts/oracle/components.doc
2  LIBCACHE             LIBCACHE (kgl, kql)

I’m guessing by the name, that the last one should be responsible for pining the PL/SQL object in library cache for the execution time.

Based on this I used a very simple DTrace script to check the execution time and number of executions of each function:

1root@skiper:~# cat plsql_proc_fun_time.d
2#!/usr/sbin/dtrace -s
3#pragma D option quiet
4 
5uint64_t        kpoTime;
6uint64_t        plsqlrunTime;
7uint64_t        kglpnpTime;
8uint64_t        contextTime;
9 
10BEGIN
11{
12  printf("%s\n","START PROFILING kgiPinObject, plsql_run, kglpnpi,kglPinSetContext execution times");
13}
14 
15pid$1::kgiPinObject:entry
16/pid==$1/
17{
18  self->t=timestamp;
19}
20pid$1::kgiPinObject:return
21/pid==$1 && self->t>0/
22{
23  kpoTime+=timestamp-self->t;
24  self->t=0;
25  @stackKgi[ustack()]=count();
26  kgiPinObjectExec++;
27}
28pid$1::plsql_run:entry
29/pid==$1/
30{
31  self->pt=timestamp;
32}
33pid$1::plsql_run:return
34/pid==$1 && self->pt>0/
35{
36  plsqlrunTime+=timestamp-self->pt;
37  self->pt=0;
38  plsql_runExec++;
39}
40pid$1::kglpnp:entry
41/pid==$1/
42{
43  self->kt=timestamp;
44}
45pid$1::kglpnp:return
46/pid==$1 && self->kt>0/
47{
48  kglpnpTime+=timestamp-self->kt;
49  self->kt=0;
50  @stackKgl[ustack()]=count();
51  kglpnpExec++;
52}
53pid$1::kglPinSetContext:entry
54/pid==$1/
55{
56  self->ct=timestamp;
57}
58pid$1::kglPinSetContext:return
59/pid==$1 && self->ct>0/
60{
61  contextTime+=timestamp-self->ct;
62  self->ct=0;
63  @stackCtx[ustack()]=count();
64  kglPinSetContextExec++;
65}
66 
67END
68{
69  printa(@stackKgi);
70  printa(@stackKgl);
71  printa(@stackCtx);
72  printf("kgiPinObject executed %d times, waited in ms: %d\n",kgiPinObjectExec,kpoTime/1000000);
73  printf("kglpnp executed %d times, waited in ms: %d\n",kglpnpExec, kglpnpTime/1000000);
74  printf("kglPinSetContext executed %d times, waited in ms: %d\n",kglPinSetContextExec, contextTime/1000000);
75  printf("plsql_run executed %d times, waited in ms: %d\n",plsql_runExec, plsqlrunTime/1000000);
76  sm=kpoTime+kglpnpTime+contextTime;
77  printf("kgiPinObject+kglpnp+kglPinSetContext = %d ms \n",sm/1000000);
78}

Please notice, that this script will seriously affect the execution times, because monitoring can change the monitored object 😉

So right now the execution time for package looks like this:

1SQL> exec pkg_test_calls.by2_proc
2 
3PL/SQL procedure successfully completed.
4 
5Elapsed: 00:00:00.18
6SQL>

And this is the output of my Dtrace script:

1root@skiper:~# ./plsql_proc_fun_time.d 3450
2START PROFILING kgiPinObject, plsql_run, kglpnpi,kglPinSetContext execution times
3^C
4 
5 
6              oracle`kgiPinObject+0xb8
7              oracle`pfrinstr_XCAL_SCAL+0x87
8              oracle`pfrrun_no_tool+0x12a
9              oracle`pfrrun+0x4c0
10              oracle`plsql_run+0x288
11              oracle`peicnt+0x15f
12              oracle`kkxexe+0x2f3
13              oracle`opiexe+0x32c3
14              oracle`kpoal8+0xa3d
15              oracle`opiodr+0x433
16              oracle`ttcpip+0x593
17              oracle`opitsk+0x6cc
18              oracle`opiino+0x3c5
19              oracle`opiodr+0x433
20              oracle`opidrv+0x33b
21              oracle`sou2o+0x7a
22              oracle`opimai_real+0x265
23              oracle`ssthrdmain+0x22e
24              oracle`main+0xa4
25              oracle`0x6e80c2b
26                1
27 
28 
29              oracle`kglpnp+0x140
30              oracle`pfri7_inst_body_common+0x88
31              oracle`pfri3_inst_body+0x43
32              oracle`pfrrun+0x685
33              oracle`plsql_run+0x288
34              oracle`peicnt+0x15f
35              oracle`kkxexe+0x2f3
36              oracle`opiexe+0x32c3
37              oracle`kpoal8+0xa3d
38              oracle`opiodr+0x433
39              oracle`ttcpip+0x593
40              oracle`opitsk+0x6cc
41              oracle`opiino+0x3c5
42              oracle`opiodr+0x433
43              oracle`opidrv+0x33b
44              oracle`sou2o+0x7a
45              oracle`opimai_real+0x265
46              oracle`ssthrdmain+0x22e
47              oracle`main+0xa4
48              oracle`0x6e80c2b
49                1
50 
51              oracle`kglpnp+0x140
52              oracle`pfri8_inst_spec+0x83
53              oracle`pfri1_inst_spec+0x43
54              oracle`pfrrun+0x5e2
55              oracle`plsql_run+0x288
56              oracle`peicnt+0x15f
57              oracle`kkxexe+0x2f3
58              oracle`opiexe+0x32c3
59              oracle`kpoal8+0xa3d
60              oracle`opiodr+0x433
61              oracle`ttcpip+0x593
62              oracle`opitsk+0x6cc
63              oracle`opiino+0x3c5
64              oracle`opiodr+0x433
65              oracle`opidrv+0x33b
66              oracle`sou2o+0x7a
67              oracle`opimai_real+0x265
68              oracle`ssthrdmain+0x22e
69              oracle`main+0xa4
70              oracle`0x6e80c2b
71                1
72 
73              oracle`kglpnp+0x140
74              oracle`pevm_CHK_INST+0x49
75              oracle`pfrinstr_XCAL_SCAL+0x87
76              oracle`pfrrun_no_tool+0x12a
77              oracle`pfrrun+0x4c0
78              oracle`plsql_run+0x288
79              oracle`peicnt+0x15f
80              oracle`kkxexe+0x2f3
81              oracle`opiexe+0x32c3
82              oracle`kpoal8+0xa3d
83              oracle`opiodr+0x433
84              oracle`ttcpip+0x593
85              oracle`opitsk+0x6cc
86              oracle`opiino+0x3c5
87              oracle`opiodr+0x433
88              oracle`opidrv+0x33b
89              oracle`sou2o+0x7a
90              oracle`opimai_real+0x265
91              oracle`ssthrdmain+0x22e
92              oracle`main+0xa4
93                1
94 
95              oracle`kglpnp+0x140
96              oracle`pfri7_inst_body_common+0x118
97              oracle`pfri3_inst_body+0x43
98              oracle`pfrrun+0x685
99              oracle`plsql_run+0x288
100              oracle`peicnt+0x15f
101              oracle`kkxexe+0x2f3
102              oracle`opiexe+0x32c3
103              oracle`kpoal8+0xa3d
104              oracle`opiodr+0x433
105              oracle`ttcpip+0x593
106              oracle`opitsk+0x6cc
107              oracle`opiino+0x3c5
108              oracle`opiodr+0x433
109              oracle`opidrv+0x33b
110              oracle`sou2o+0x7a
111              oracle`opimai_real+0x265
112              oracle`ssthrdmain+0x22e
113              oracle`main+0xa4
114              oracle`0x6e80c2b
115                1
116 
117 
118              oracle`kglPinSetContext+0x76
119              oracle`pevm_CHK_INST+0x49
120              oracle`pfrinstr_XCAL_SCAL+0x87
121              oracle`pfrrun_no_tool+0x12a
122              oracle`pfrrun+0x4c0
123              oracle`plsql_run+0x288
124              oracle`peicnt+0x15f
125              oracle`kkxexe+0x2f3
126              oracle`opiexe+0x32c3
127              oracle`kpoal8+0xa3d
128              oracle`opiodr+0x433
129              oracle`ttcpip+0x593
130              oracle`opitsk+0x6cc
131              oracle`opiino+0x3c5
132              oracle`opiodr+0x433
133              oracle`opidrv+0x33b
134              oracle`sou2o+0x7a
135              oracle`opimai_real+0x265
136              oracle`ssthrdmain+0x22e
137              oracle`main+0xa4
138                1
139 
140              oracle`kglPinSetContext+0x76
141              oracle`kgiind+0x431
142              oracle`pfri8_inst_spec+0x83
143              oracle`pfri1_inst_spec+0x43
144              oracle`pfrrun+0x5e2
145              oracle`plsql_run+0x288
146              oracle`peicnt+0x15f
147              oracle`kkxexe+0x2f3
148              oracle`opiexe+0x32c3
149              oracle`kpoal8+0xa3d
150              oracle`opiodr+0x433
151              oracle`ttcpip+0x593
152              oracle`opitsk+0x6cc
153              oracle`opiino+0x3c5
154              oracle`opiodr+0x433
155              oracle`opidrv+0x33b
156              oracle`sou2o+0x7a
157              oracle`opimai_real+0x265
158              oracle`ssthrdmain+0x22e
159              oracle`main+0xa4
160                1
161 
162              oracle`kglPinSetContext+0x76
163              oracle`kglpndl+0xd23
164              oracle`kss_del_cb+0x19d
165              oracle`kssdel+0xf2
166              oracle`kssdch_int+0x326
167              oracle`ksudlc+0xf6
168              oracle`kss_del_cb+0x106
169              oracle`kssdel+0xf2
170              oracle`ksupop+0x23a
171              oracle`opiodr+0x4b1
172              oracle`ttcpip+0x593
173              oracle`opitsk+0x6cc
174              oracle`opiino+0x3c5
175              oracle`opiodr+0x433
176              oracle`opidrv+0x33b
177              oracle`sou2o+0x7a
178              oracle`opimai_real+0x265
179              oracle`ssthrdmain+0x22e
180              oracle`main+0xa4
181              oracle`0x6e80c2b
182                2
183kgiPinObject executed 1 times, waited in ms: 0
184kglpnp executed 4 times, waited in ms: 0
185kglPinSetContext executed 4 times, waited in ms: 0
186plsql_run executed 1 times, waited in ms: 172
187kgiPinObject+kglpnp+kglPinSetContext = 0 ms

And this is the execution time of procedure call:

1SQL> exec proc_by2
2 
3PL/SQL procedure successfully completed.
4 
5Elapsed: 00:00:24.38

And the result of my DTrace script for this execution:

1root@skiper:~# ./plsql_proc_fun_time.d 3450
2START PROFILING kgiPinObject, plsql_run, kglpnpi,kglPinSetContext execution times
3^C
4 
5 
6              oracle`kgiPinObject+0xb8
7              oracle`pfrinstr_XCAL_SCAL+0x87
8              oracle`pfrrun_no_tool+0x12a
9              oracle`pfrrun+0x4c0
10              oracle`plsql_run+0x288
11              oracle`peicnt+0x15f
12              oracle`kkxexe+0x2f3
13              oracle`opiexe+0x32c3
14              oracle`kpoal8+0xa3d
15              oracle`opiodr+0x433
16              oracle`ttcpip+0x593
17              oracle`opitsk+0x6cc
18              oracle`opiino+0x3c5
19              oracle`opiodr+0x433
20              oracle`opidrv+0x33b
21              oracle`sou2o+0x7a
22              oracle`opimai_real+0x265
23              oracle`ssthrdmain+0x22e
24              oracle`main+0xa4
25              oracle`0x6e80c2b
26          1000002
27 
28 
29              oracle`kglpnp+0x140
30              oracle`pfri7_inst_body_common+0x88
31              oracle`pfri3_inst_body+0x43
32              oracle`pfrrun+0x685
33              oracle`plsql_run+0x288
34              oracle`peicnt+0x15f
35              oracle`kkxexe+0x2f3
36              oracle`opiexe+0x32c3
37              oracle`kpoal8+0xa3d
38              oracle`opiodr+0x433
39              oracle`ttcpip+0x593
40              oracle`opitsk+0x6cc
41              oracle`opiino+0x3c5
42              oracle`opiodr+0x433
43              oracle`opidrv+0x33b
44              oracle`sou2o+0x7a
45              oracle`opimai_real+0x265
46              oracle`ssthrdmain+0x22e
47              oracle`main+0xa4
48              oracle`0x6e80c2b
49                2
50 
51              oracle`kglpnp+0x140
52              oracle`pfri8_inst_spec+0x83
53              oracle`pfri1_inst_spec+0x43
54              oracle`pfrrun+0x5e2
55              oracle`plsql_run+0x288
56              oracle`peicnt+0x15f
57              oracle`kkxexe+0x2f3
58              oracle`opiexe+0x32c3
59              oracle`kpoal8+0xa3d
60              oracle`opiodr+0x433
61              oracle`ttcpip+0x593
62              oracle`opitsk+0x6cc
63              oracle`opiino+0x3c5
64              oracle`opiodr+0x433
65              oracle`opidrv+0x33b
66              oracle`sou2o+0x7a
67              oracle`opimai_real+0x265
68              oracle`ssthrdmain+0x22e
69              oracle`main+0xa4
70              oracle`0x6e80c2b
71                2
72 
73              oracle`kglpnp+0x140
74              oracle`pevm_CHK_INST+0x49
75              oracle`pfrinstr_XCAL_SCAL+0x87
76              oracle`pfrrun_no_tool+0x12a
77              oracle`pfrrun+0x4c0
78              oracle`plsql_run+0x288
79              oracle`peicnt+0x15f
80              oracle`kkxexe+0x2f3
81              oracle`opiexe+0x32c3
82              oracle`kpoal8+0xa3d
83              oracle`opiodr+0x433
84              oracle`ttcpip+0x593
85              oracle`opitsk+0x6cc
86              oracle`opiino+0x3c5
87              oracle`opiodr+0x433
88              oracle`opidrv+0x33b
89              oracle`sou2o+0x7a
90              oracle`opimai_real+0x265
91              oracle`ssthrdmain+0x22e
92              oracle`main+0xa4
93          1000002
94 
95 
96              oracle`kglPinSetContext+0x76
97              oracle`kgiind+0x431
98              oracle`pfri8_inst_spec+0x83
99              oracle`pfri1_inst_spec+0x43
100              oracle`pfrrun+0x5e2
101              oracle`plsql_run+0x288
102              oracle`peicnt+0x15f
103              oracle`kkxexe+0x2f3
104              oracle`opiexe+0x32c3
105              oracle`kpoal8+0xa3d
106              oracle`opiodr+0x433
107              oracle`ttcpip+0x593
108              oracle`opitsk+0x6cc
109              oracle`opiino+0x3c5
110              oracle`opiodr+0x433
111              oracle`opidrv+0x33b
112              oracle`sou2o+0x7a
113              oracle`opimai_real+0x265
114              oracle`ssthrdmain+0x22e
115              oracle`main+0xa4
116                1
117 
118              oracle`kglPinSetContext+0x76
119              oracle`kglpndl+0xd23
120              oracle`kss_del_cb+0x19d
121              oracle`kssdel+0xf2
122              oracle`kssdch_int+0x326
123              oracle`ksudlc+0xf6
124              oracle`kss_del_cb+0x106
125              oracle`kssdel+0xf2
126              oracle`ksupop+0x23a
127              oracle`opiodr+0x4b1
128              oracle`ttcpip+0x593
129              oracle`opitsk+0x6cc
130              oracle`opiino+0x3c5
131              oracle`opiodr+0x433
132              oracle`opidrv+0x33b
133              oracle`sou2o+0x7a
134              oracle`opimai_real+0x265
135              oracle`ssthrdmain+0x22e
136              oracle`main+0xa4
137              oracle`0x6e80c2b
138                2
139 
140              oracle`kglPinSetContext+0x76
141              oracle`pevm_CHK_INST+0x49
142              oracle`pfrinstr_XCAL_SCAL+0x87
143              oracle`pfrrun_no_tool+0x12a
144              oracle`pfrrun+0x4c0
145              oracle`plsql_run+0x288
146              oracle`peicnt+0x15f
147              oracle`kkxexe+0x2f3
148              oracle`opiexe+0x32c3
149              oracle`kpoal8+0xa3d
150              oracle`opiodr+0x433
151              oracle`ttcpip+0x593
152              oracle`opitsk+0x6cc
153              oracle`opiino+0x3c5
154              oracle`opiodr+0x433
155              oracle`opidrv+0x33b
156              oracle`sou2o+0x7a
157              oracle`opimai_real+0x265
158              oracle`ssthrdmain+0x22e
159              oracle`main+0xa4
160          1000001
161kgiPinObject executed 1000002 times, waited in ms: 19482
162kglpnp executed 1000006 times, waited in ms: 3532
163kglPinSetContext executed 1000004 times, waited in ms: 3676
164plsql_run executed 1 times, waited in ms: 24380
165kgiPinObject+kglpnp+kglPinSetContext = 26692 ms

As you can see, the numbers are accurate enough to make some conclusions: execution time of function kgiPinObject is almost 80% of the whole execution time!

So the reason why PL/SQL packages are faster in subsequent invocations of other subprograms is not the "disk I/O", but penalty due to the need to pin/lock the foreign object in Library Cache each time, when I want to execute it.

So the practical knowledge "the lightning kills" did not change, but now at least I know, that it’s not because Zeus is mad at me 😉

P.S.
If you want to meet true experts in PL/SQL, you have to visit Poland, 7/8 OCTOBER 2016 at POUG (Pint with Oracle Users Group). Details can be found in here: http://poug.org/en/
Name of the function, which executes kgiPinObject will give -10% for the ticket 😉


Contact us

Database Whisperers sp. z o. o. sp. k.
al. Jerozolimskie 200, 3rd floor, room 342
02-486 Warszawa
NIP: 5272744987
REGON:362524978
+48 508 943 051
+48 661 966 009
info@ora-600.pl

Newsletter Sign up to be updated