Let’s check, why PRAGMA UDF makes execution faster, then regular function.
C function responsible for calling a PL/SQL code from SQL is called "plsql_run":
1 | [oracle@rico bin]$ nm oracle | grep plsql_run |
2 | 000000000ce1c090 T plsql_run |
Now let’s try to create a new HR session and create a simple function which will multiple values by 2:
5 | 2 function by2(x number) return number is |
From other terminal I will connect to the HR session with GDB:
1 | [root@rico ~]# gdb -p 10330 |
2 | GNU gdb (GDB) Red Hat Enterprise Linux (7.2-83.el6) |
3 | Copyright (C) 2010 Free Software Foundation, Inc. |
4 | License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html> |
5 | This is free software: you are free to change and redistribute it. |
6 | There is NO WARRANTY, to the extent permitted by law. Type "show copying" |
7 | and "show warranty" for details. |
8 | This GDB was configured as "x86_64-redhat-linux-gnu". |
9 | For bug reporting instructions, please see: |
10 | <http://www.gnu.org/software/gdb/bugs/>. |
11 | Attaching to process 10330 |
12 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/bin/oracle...(no debugging symbols found)...done. |
13 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libodm12.so...(no debugging symbols found)...done. |
14 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libodm12.so |
15 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libcell12.so...done. |
16 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libcell12.so |
17 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxp12.so...(no debugging symbols found)...done. |
18 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxp12.so |
19 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskjcx12.so...done. |
20 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskjcx12.so |
21 | Reading symbols from /lib64/librt.so.1...Reading symbols from /usr/lib/debug/lib64/librt-2.12.so.debug...done. |
23 | Loaded symbols for /lib64/librt.so.1 |
24 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libmql1.so...(no debugging symbols found)...done. |
25 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libmql1.so |
26 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libipc1.so...(no debugging symbols found)...done. |
27 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libipc1.so |
28 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libclsra12.so...done. |
29 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libclsra12.so |
30 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libdbcfg12.so...(no debugging symbols found)...done. |
31 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libdbcfg12.so |
32 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libhasgen12.so...done. |
33 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libhasgen12.so |
34 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxn2.so...(no debugging symbols found)...done. |
35 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxn2.so |
36 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocr12.so...done. |
37 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocr12.so |
38 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrb12.so...done. |
39 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrb12.so |
40 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrutl12.so...done. |
41 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrutl12.so |
42 | Reading symbols from /lib64/libaio.so.1...Reading symbols from /usr/lib/debug/lib64/libaio.so.1.0.1.debug...done. |
44 | Loaded symbols for /lib64/libaio.so.1 |
45 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libons.so...(no debugging symbols found)...done. |
46 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libons.so |
47 | Reading symbols from /lib64/libdl.so.2...Reading symbols from /usr/lib/debug/lib64/libdl-2.12.so.debug...done. |
49 | Loaded symbols for /lib64/libdl.so.2 |
50 | Reading symbols from /lib64/libm.so.6...Reading symbols from /usr/lib/debug/lib64/libm-2.12.so.debug...done. |
52 | Loaded symbols for /lib64/libm.so.6 |
53 | Reading symbols from /lib64/libpthread.so.0...Reading symbols from /usr/lib/debug/lib64/libpthread-2.12.so.debug...done. |
54 | [Thread debugging using libthread_db enabled] |
56 | Loaded symbols for /lib64/libpthread.so.0 |
57 | Reading symbols from /lib64/libnsl.so.1...Reading symbols from /usr/lib/debug/lib64/libnsl-2.12.so.debug...done. |
59 | Loaded symbols for /lib64/libnsl.so.1 |
60 | Reading symbols from /lib64/libc.so.6...Reading symbols from /usr/lib/debug/lib64/libc-2.12.so.debug...done. |
62 | Loaded symbols for /lib64/libc.so.6 |
63 | Reading symbols from /lib64/ld-linux-x86-64.so.2...Reading symbols from /usr/lib/debug/lib64/ld-2.12.so.debug...done. |
65 | Loaded symbols for /lib64/ld-linux-x86-64.so.2 |
66 | Reading symbols from /usr/lib64/libnuma.so.1...Reading symbols from /usr/lib/debug/usr/lib64/libnuma.so.1.debug...done. |
68 | Loaded symbols for /usr/lib64/libnuma.so.1 |
69 | Reading symbols from /lib64/libnss_files.so.2...Reading symbols from /usr/lib/debug/lib64/libnss_files-2.12.so.debug...done. |
71 | Loaded symbols for /lib64/libnss_files.so.2 |
72 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libnque12.so...(no debugging symbols found)...done. |
73 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libnque12.so |
74 | Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libshpkavx12.so...(no debugging symbols found)...done. |
75 | Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libshpkavx12.so |
77 | warning: no loadable sections found in added symbol-file system-supplied DSO at 0x7ffc971db000 |
78 | 0x000000390c00e7c0 in __read_nocancel () at ../sysdeps/unix/syscall-template.S:82 |
79 | 82 T_PSEUDO (SYSCALL_SYMBOL, SYSCALL_NAME, SYSCALL_NARGS) |
and I will set the breakpoint for plsql_run function
2 | Breakpoint 1 at 0xce1c090 |
4 | Num Type Disp Enb Address What |
5 | 1 breakpoint keep y 0x000000000ce1c090 <plsql_run> |
7 | Type commands for breakpoint(s) 1, one per line. |
8 | End with a line saying just "end". |
12 | Num Type Disp Enb Address What |
13 | 1 breakpoint keep y 0x000000000ce1c090 <plsql_run> |
15 | (gdb) set pagination off |
In the HR session, I will use my function on the table "EMPLOYEES".
1 | SQL> select by2(salary) from employees; |
In the gdb we can see, that the breakpoint was hit 107 times – this is related to the number of rows in the table.
And what will happen if we add "PRAGMA UDF" syntax to our little function?
5 | 2 function by2(x number) return number is |
14 | SQL> select by2(salary) from employees; |
The gdb shows a funny thing:
2 | Num Type Disp Enb Address What |
3 | 2 breakpoint keep y 0x000000000ce1c090 <plsql_run> |
4 | breakpoint already hit 9 times |
So why, the function was called only 9 times?
The tkprof shows something that can be interesting:
6 | call count cpu elapsed disk query current rows |
7 | ------- ------ -------- ---------- ---------- ---------- ---------- ---------- |
8 | Parse 1 0.00 0.00 0 0 0 0 |
9 | Execute 1 0.00 0.00 0 0 0 0 |
10 | Fetch 9 0.00 0.00 0 15 0 107 |
11 | ------- ------ -------- ---------- ---------- ---------- ---------- ---------- |
12 | total 11 0.00 0.01 0 15 0 107 |
There was 9 fetches! Is it possible that the function was called once for each fetch? We can make sure by changing the arraysize in SQL*Plus
2 | SQL> select by2(salary) from employees; |
The gdb shows three plsql_run executions:
2 | Num Type Disp Enb Address What |
3 | 3 breakpoint keep y 0x000000000ce1c090 <plsql_run> |
4 | breakpoint already hit 3 times |
And the tkprof also:
6 | call count cpu elapsed disk query current rows |
7 | ------- ------ -------- ---------- ---------- ---------- ---------- ---------- |
8 | Parse 1 0.00 0.00 0 0 0 0 |
9 | Execute 1 0.00 0.00 0 0 0 0 |
10 | Fetch 3 0.00 0.00 0 9 0 107 |
11 | ------- ------ -------- ---------- ---------- ---------- ---------- ---------- |
12 | total 5 0.00 0.00 0 9 0 107 |
So the fetch size can have actual impact on PRAGMA UDF functionality.
Let’s see what will happen from inside of the PL/SQL – without throwing the results to the client:
5 | 2 type t_nums is table of number index by pls_integer; |
8 | 5 select by2(salary) bulk collect into v_nums |
13 | PL/SQL procedure successfully completed. |
gdb shows two plsql_run hits – one for anonymous block and one for the "BY2″ function.