Oracle 12c: PRAGMA UDF – the truth


29.10.2015
by Kamil Stawiarski

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
2000000000ce1c090 T plsql_run

Now let’s try to create a new HR session and create a simple function which will multiple values by 2:

1SQL> conn hr/hr
2Connected.
3SQL> get by2
4  1  create or replace
5  2  function by2(x number) return number is
6  3  begin
7  4    return x*2;
8  5* end;
9SQL> /
10 
11Function created.

From other terminal I will connect to the HR session with GDB:

1[root@rico ~]# gdb -p 10330
2GNU gdb (GDB) Red Hat Enterprise Linux (7.2-83.el6)
3Copyright (C) 2010 Free Software Foundation, Inc.
4License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>
5This is free software: you are free to change and redistribute it.
6There is NO WARRANTY, to the extent permitted by law.  Type "show copying"
7and "show warranty" for details.
8This GDB was configured as "x86_64-redhat-linux-gnu".
9For bug reporting instructions, please see:
10<http://www.gnu.org/software/gdb/bugs/>.
11Attaching to process 10330
12Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/bin/oracle...(no debugging symbols found)...done.
13Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libodm12.so...(no debugging symbols found)...done.
14Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libodm12.so
15Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libcell12.so...done.
16Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libcell12.so
17Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxp12.so...(no debugging symbols found)...done.
18Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxp12.so
19Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskjcx12.so...done.
20Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskjcx12.so
21Reading symbols from /lib64/librt.so.1...Reading symbols from /usr/lib/debug/lib64/librt-2.12.so.debug...done.
22done.
23Loaded symbols for /lib64/librt.so.1
24Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libmql1.so...(no debugging symbols found)...done.
25Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libmql1.so
26Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libipc1.so...(no debugging symbols found)...done.
27Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libipc1.so
28Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libclsra12.so...done.
29Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libclsra12.so
30Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libdbcfg12.so...(no debugging symbols found)...done.
31Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libdbcfg12.so
32Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libhasgen12.so...done.
33Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libhasgen12.so
34Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxn2.so...(no debugging symbols found)...done.
35Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libskgxn2.so
36Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocr12.so...done.
37Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocr12.so
38Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrb12.so...done.
39Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrb12.so
40Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrutl12.so...done.
41Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libocrutl12.so
42Reading symbols from /lib64/libaio.so.1...Reading symbols from /usr/lib/debug/lib64/libaio.so.1.0.1.debug...done.
43done.
44Loaded symbols for /lib64/libaio.so.1
45Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libons.so...(no debugging symbols found)...done.
46Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libons.so
47Reading symbols from /lib64/libdl.so.2...Reading symbols from /usr/lib/debug/lib64/libdl-2.12.so.debug...done.
48done.
49Loaded symbols for /lib64/libdl.so.2
50Reading symbols from /lib64/libm.so.6...Reading symbols from /usr/lib/debug/lib64/libm-2.12.so.debug...done.
51done.
52Loaded symbols for /lib64/libm.so.6
53Reading 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]
55done.
56Loaded symbols for /lib64/libpthread.so.0
57Reading symbols from /lib64/libnsl.so.1...Reading symbols from /usr/lib/debug/lib64/libnsl-2.12.so.debug...done.
58done.
59Loaded symbols for /lib64/libnsl.so.1
60Reading symbols from /lib64/libc.so.6...Reading symbols from /usr/lib/debug/lib64/libc-2.12.so.debug...done.
61done.
62Loaded symbols for /lib64/libc.so.6
63Reading symbols from /lib64/ld-linux-x86-64.so.2...Reading symbols from /usr/lib/debug/lib64/ld-2.12.so.debug...done.
64done.
65Loaded symbols for /lib64/ld-linux-x86-64.so.2
66Reading symbols from /usr/lib64/libnuma.so.1...Reading symbols from /usr/lib/debug/usr/lib64/libnuma.so.1.debug...done.
67done.
68Loaded symbols for /usr/lib64/libnuma.so.1
69Reading symbols from /lib64/libnss_files.so.2...Reading symbols from /usr/lib/debug/lib64/libnss_files-2.12.so.debug...done.
70done.
71Loaded symbols for /lib64/libnss_files.so.2
72Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libnque12.so...(no debugging symbols found)...done.
73Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libnque12.so
74Reading symbols from /u01/app/oracle/product/12.1.0/dbhome_1/lib/libshpkavx12.so...(no debugging symbols found)...done.
75Loaded symbols for /u01/app/oracle/product/12.1.0/dbhome_1/lib/libshpkavx12.so
76 
77warning: no loadable sections found in added symbol-file system-supplied DSO at 0x7ffc971db000
780x000000390c00e7c0 in __read_nocancel () at ../sysdeps/unix/syscall-template.S:82
7982  T_PSEUDO (SYSCALL_SYMBOL, SYSCALL_NAME, SYSCALL_NARGS)
80(gdb)

and I will set the breakpoint for plsql_run function

1(gdb) b plsql_run
2Breakpoint 1 at 0xce1c090
3(gdb) info breakpoint
4Num     Type           Disp Enb Address            What
51       breakpoint     keep y   0x000000000ce1c090 <plsql_run>
6(gdb) command 1
7Type commands for breakpoint(s) 1, one per line.
8End with a line saying just "end".
9>c
10>end
11(gdb) info b
12Num     Type           Disp Enb Address            What
131       breakpoint     keep y   0x000000000ce1c090 <plsql_run>
14        c
15(gdb) set pagination off
16(gdb) c
17Continuing.

In the HR session, I will use my function on the table "EMPLOYEES".

1SQL> select by2(salary) from employees;
2(...output removed...)
3107 rows selected.
4 
5SQL>

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?

1SQL> ed
2Wrote file afiedt.buf
3 
4  1  create or replace
5  2  function by2(x number) return number is
6  3  pragma udf;
7  4  begin
8  5    return x*2;
9  6* end;
10SQL> /
11 
12Function created.
13 
14SQL> select by2(salary) from employees;
15(...output removed...)
16107 rows selected.
17 
18SQL>

The gdb shows a funny thing:

1(gdb) info b
2Num     Type           Disp Enb Address            What
32       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:

1select by2(salary)
2from
3 employees
4 
5 
6call     count       cpu    elapsed       disk      query    current        rows
7------- ------  -------- ---------- ---------- ---------- ----------  ----------
8Parse        1      0.00       0.00          0          0          0           0
9Execute      1      0.00       0.00          0          0          0           0
10Fetch        9      0.00       0.00          0         15          0         107
11------- ------  -------- ---------- ---------- ---------- ----------  ----------
12total       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

1SQL> set arraysize 66
2SQL> select by2(salary) from employees;
3(...output removed...)
4107 rows selected.
5 
6SQL>

The gdb shows three plsql_run executions:

1(gdb) info b
2Num     Type           Disp Enb Address            What
33       breakpoint     keep y   0x000000000ce1c090 <plsql_run>
4    breakpoint already hit 3 times
5        c

And the tkprof also:

1select by2(salary)
2from
3 employees
4 
5 
6call     count       cpu    elapsed       disk      query    current        rows
7------- ------  -------- ---------- ---------- ---------- ----------  ----------
8Parse        1      0.00       0.00          0          0          0           0
9Execute      1      0.00       0.00          0          0          0           0
10Fetch        3      0.00       0.00          0          9          0         107
11------- ------  -------- ---------- ---------- ---------- ----------  ----------
12total        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:

1SQL> ed
2Wrote file afiedt.buf
3 
4  1  declare
5  2 type t_nums is table of number index by pls_integer;
6  3 v_nums t_nums;
7  4  begin
8  5 select by2(salary) bulk collect into v_nums
9  6 from employees;
10  7* end;
11SQL> /
12 
13PL/SQL procedure successfully completed.

gdb shows two plsql_run hits – one for anonymous block and one for the "BY2″ function.


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