ORACLE/Performance Tuning

오라클 19c FULL TABLE SCAN (FTS)시 direct path read 이벤트 관련

내맘대로긍정 2024. 3. 21. 20:25

OS환경 : Oracle Linux 7.6 (64bit)

 

DB 환경 : Oracle Database 19.3.0.0

 

방법 : 오라클 19c FULL TABLE SCAN (FTS)시 direct path read 이벤트 관련

이전 테스트(group by 테스트) 진행중 테이블 full 스캔을 할때 direct path read 가 발생하는것을 보고 궁금해서 찾아봄
참고 ( 오라클 19c group by 컬럼 나열 및 집계함수 적용 성능 비교 https://positivemh.tistory.com/1034 )

 

 

오라클에서 기본적으로 Direct Path I/O 는 아래 경우에만 동작함
- CTAS 작업
- APPEND 힌트를 사용한 insert 작업
- PARALLEL 힌트를 사용한 병렬쿼리 FULL 스캔
- PARALLEL_INDEX 힌트를 사용한 병렬쿼리 INDEX 스캔
- PARALLEL 힌트를 사용한 병렬 DML
- TEMP를 읽고 쓰는 작업
- NOCACHE 옵션으로 LOB 컬럼 읽는 작업
- Datapump 사용시
- 전통 EXP/IMP 중 Direct=Y 옵션 사용시
- SQL*Loader 중 Direct=Y 옵션 사용시

 

 

이 경우가 아닌 일반 테이블을 FULL 스캔 할때는 
테이블 블록이 데이터 파일에서 버퍼 캐시로 읽혀지기 때문에 db file scattered read 이벤트가 발생해야함

그런데 일반 테이블을 FULL 스캔(병렬X 직렬) 할 때 10046 트레이스를 보니 db file scattered read 가 아닌 direct path read 이벤트가 발생하는걸 발견함

 

 

10046 테스트
샘플 테이블 생성

1
2
3
4
5
6
7
8
9
SQL>
conn / as sysdba
drop table test_emp;
create table big_emp as select * from emp connect by level <= 6;
select count(*) from test_emp;
 
  COUNT(*)
----------
   8108730

 

 

버퍼캐시 플러쉬

1
2
3
SQL> alter system flush buffer_cache;
 
System altered.

 

 

테이블 풀스캔 후 10046 트레이스 확인
(오라클 19c 10046 trace 및 tkprof 자동수행 스크립트 https://positivemh.tistory.com/960 활용)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
$ sh 10046.sh
$ cat /ORA19/app/oracle/diag/rdbms/oracle19/oracle19/trace/oracle19_ora_37734_t1_tkprof.txt
********************************************************************************
 
SQL ID: fqhvntk5d9gbr Plan Hash: 2674116386
 
select count(*)
from
 test_emp
 
 
call     count       cpu    elapsed       disk      query    current        rows
------- ------  -------- ---------- ---------- ---------- ----------  ----------
Parse        1      0.00       0.00          0          0          0           0
Execute      1      0.00       0.00          0          0          0           0
Fetch        1      0.10       0.10      47730      47733          0           1
------- ------  -------- ---------- ---------- ---------- ----------  ----------
total        3      0.10       0.10      47730      47733          0           1
 
Misses in library cache during parse: 1
Optimizer mode: ALL_ROWS
Parsing user id: 78
Number of plan statistics captured: 1
 
Rows (1st) Rows (avg) Rows (max)  Row Source Operation
---------- ---------- ----------  ---------------------------------------------------
         1          1          1  SORT AGGREGATE (cr=47733 pr=47730 pw=0 time=104406 us starts=1)
   8108730    8108730    8108730   TABLE ACCESS FULL TEST_EMP (cr=47733 pr=47730 pw=0 time=110225 us starts=1 cost=13110 size=0 card=8108730)
 
 
Elapsed times include waiting on following events:
  Event waited on                             Times   Max. Wait  Total Waited
  ----------------------------------------   Waited  ----------  ------------
  PGA memory operation                            5        0.00          0.00
  SQL*Net message to client                       1        0.00          0.00
  Disk file operations I/O                        1        0.00          0.00
  db file sequential read                         1        0.00          0.00
  direct path read                             6003        0.00          0.03 ***
  SQL*Net message from client                     1        0.00          0.00
*****************************************************1***************************

direct path read 이벤트가 많이 발생함

 

 

원인은 11g 부터 도입된 _serial_direct_read 파라미터 였음
_serial_direct_read 파라미터는 FULL 스캔시(병렬쿼리가 아닌 직렬 방식) direct path read를 가능하게 제어 해주는 파라미터임
파라미터 값이 auto(기본값) 일 경우 _small_table_threshold 파라미터 값보다 대상 테이블의 blocks(dba_tables) 값이 크다면 Direct Path I/O가 동작함
(여기서 참조하는 blocks 갯수는 _direct_read_decision_statistics_driven 파라미터와도 연관이 있다고함 기본값은 true임)
파라미터 값이 never 일 경우 절대 direct path read를 수행하지 않음
이외에 true, false, always 옵션이 존재함
참고로 _small_table_threshold 파라미터 기본값은 _db_block_buffers 파라미터의 2%임

 

 

관련 히든 파라미터 조회

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
SQL>
set lines 200 pages 1000
col desc for a70
col name for a40
col current_value for a15
col default_value for a15
col default_t_f for a15
select
ksppinm "name",
ksppstvl "current_value",
b.ksppstdfl "default_value",
b.ksppstdf "default_t_f",
ksppdesc "desc"
from sys.x$ksppi a, sys.x$ksppcv b
where 1=1
and a.indx=b.indx
AND SUBSTR(a.KSPPINM, 1, 1) = '_'
and a.ksppinm in ('_serial_direct_read', '_small_table_threshold', 
'_very_large_object_threshold', '_direct_read_decision_statistics_driven',
'_db_block_buffers')
order by 1;
 
name                                     current_value   default_value   default_t_f     desc
---------------------------------------- --------------- --------------- --------------- ----------------------------------------------------------------------
_db_block_buffers                        17622           0               TRUE            Number of database blocks cached in memory: hidden parameter
_direct_read_decision_statistics_driven  TRUE            TRUE            TRUE            enable direct read decision based on optimizer statistics
_serial_direct_read                      auto            auto            TRUE            enable direct read in serial
_small_table_threshold                   352                             TRUE            lower threshold level of table size for direct reads
_very_large_object_threshold             500             500             TRUE            upper threshold level of object size for direct reads

*_db_block_buffers 값이 17622임, _small_table_threshold 값은 352임(_db_block_buffers 17622의 2%임)

 

 

파라미터에 따른 변화 테스트

테스트1. _small_table_threshold 값보다 테이블의 블록이 많은 경우

테스트2. _small_table_threshold 값보다 테이블의 블록이 적은 경우

테스트3. _serial_direct_read 값이 never인 경우

 

 

샘플 테이블 생성 후 통계 수집

1
2
3
4
5
SQL>
conn / as sysdba
drop table test_emp purge;
create table test_emp as select * from emp, dual connect by level <= 6;
exec dbms_stats.gather_table_stats('SYS', 'TEST_EMP');

 

 

블록 갯수 확인

1
2
3
4
5
6
7
8
9
SQL>
col table_name for a20
select table_name, num_rows, blocks 
from dba_tables 
where table_name = 'TEST_EMP';
 
TABLE_NAME             NUM_ROWS     BLOCKS
-------------------- ---------- ----------
TEST_EMP                8108730      50029

50029개 블록을 가지고 있음

 

 

테스트1. _small_table_threshold 값보다 테이블의 블록이 많은 경우
현재 _serial_direct_read 파라미터는 auto이고, _small_table_threshold 파라미터 값은 352임
_small_table_threshold(352) 보다 test_emp 테이블의 블록(50029)이 더 많은 상태임

 

 

버퍼캐시 플러쉬

1
2
3
SQL> alter system flush buffer_cache;
 
System altered.

 

 

test_emp 테이블 조회 후 direct path read 사용량 확인

1
2
3
4
5
6
7
8
9
10
11
12
SQL> select /*+ test1 */ count(*) from test_emp;
 
  COUNT(*)
----------
   8108730
 
select sql_id, substr(sql_text, 1, 30) "sql_text", direct_reads, buffer_gets from v$sql
where sql_text like '%test1%';
 
SQL_ID        sql_text                                                     DIRECT_READS BUFFER_GETS
------------- ------------------------------------------------------------ ------------ -----------
4f4cymkj342j2 select /*+ test1 */ count(*) f                                      50029       50049

DIRECT_READS 값이 50029임(Direct Path I/O가 작동함)

 

 

버퍼캐시 플러쉬

1
2
3
SQL> alter system flush buffer_cache;
 
System altered.

 

 

테스트2. _small_table_threshold 값보다 테이블의 블록이 적은 경우
_small_table_threshold 파라미터 50030으로 변경

1
2
3
SQL> alter session set "_small_table_threshold" = 50030;
 
Session altered.

 

 

현재 _serial_direct_read 파라미터는 auto이고, _small_table_threshold 파라미터 값은 50030임
_small_table_threshold(50030) 보다 test_emp 테이블의 블록(50029)이 더 적은 상태임

 

 

test_emp 테이블 조회 후 direct path read 사용량 확인

1
2
3
4
5
6
7
8
9
10
11
12
SQL> select /*+ test2 */ count(*) from test_emp;
 
  COUNT(*)
----------
   8108730
 
select sql_id, substr(sql_text, 1, 30) "sql_text", direct_reads, buffer_gets from v$sql
where sql_text like '%test2%';
 
SQL_ID        sql_text                                                     DIRECT_READS BUFFER_GETS
------------- ------------------------------------------------------------ ------------ -----------
a0m87v5jnf0u4 select /*+ test2 */ count(*) f                                          0       50110

DIRECT_READS 값이 0임(Direct Path I/O가 미작동함)

 

 

테스트3. _serial_direct_read 값이 never인 경우
_small_table_threshold 파라미터 초기값으로 변경 및 _serial_direct_read 파라미터 never 으로 변경

1
2
3
SQL> 
alter session set "_small_table_threshold" = 352;
alter session set "_serial_direct_read" = never;

 

 

버퍼캐시 플러쉬

1
2
3
SQL> alter system flush buffer_cache;
 
System altered.

 

 

test_emp 테이블 조회 후 direct path read 사용량 확인

1
2
3
4
5
6
7
8
9
10
11
12
SQL> select /*+ test3 */ count(*) from test_emp;
 
  COUNT(*)
----------
   8108730
 
select sql_id, substr(sql_text, 1, 30) "sql_text", direct_reads, buffer_gets from v$sql
where sql_text like '%test3%';
 
SQL_ID        sql_text                                                     DIRECT_READS BUFFER_GETS
------------- ------------------------------------------------------------ ------------ -----------
29xpytcp085n9 select /*+ test3 */ count(*) f                                          0       50042

DIRECT_READS 값이 0임(Direct Path I/O가 미작동함)
_small_table_threshold 값이 352임에도 _serial_direct_read 값이 never로 설정되어 있어서 Direct Path I/O가 미동작함

 

 

주의사항
이처럼_serial_direct_read 파라미터가 auto로 설정되어 있을때 테이블 블록이 _small_table_threshold 값보다 큰 경우 Direct Path I/O가 동작함
이 말은 쿼리에서 특정 테이블이 FULL 하는 실행계획으로 풀릴 때 쿼리 수행시마다 캐싱(버퍼캐시)을 전혀 사용하지 않고 매번 Direct Path I/O를 한다는 뜻임
쿼리가 자주 수행되는 경우라면 성능 저하를 겪을 수 있음
참고 : _small_table_threshold 파라미터값 조정을 통한 시스템 성능 향상 사례 ( https://cafe.naver.com/dbian/1673 )
또한 block cleanout 관련 이슈도 발생할 수 있음
참고 : Direct path read 와 block cleanout 그리고 latch: row cache objects ( https://engineering-skcc.github.io/oracle%20tuning/adaptive-direct-path-load/ )

 

 

결론 :
_serial_direct_read 파라미터가 auto일때
_small_table_threshold 값보다 대상 테이블의 블록이 더 많은 경우 Direct Path I/O 가 동작함
_small_table_threshold 값보다 대상 테이블의 블록이 더 적은 경우 Direct Path I/O 가 미동작함
_serial_direct_read 파라미터가 never 일때 Direct Path I/O 가 미동작함
direct path read 이벤트가 많이 발생하는 경우 반복적으로 direct path read를 사용하는 FULL 스캔 쿼리가 없는지
, _small_table_threshold 파라미터 값은 적절한지 확인이 필요함
block cleanout 이슈가 발생할 경우 원인 분석 후 변경이 많이 발생되는 테이블들을 대상으로
주기적으로 수동 cleanout 시키는 방법을 고려할 수 있음(non-direct path read 방식으로 FULL 스캔 수행)

 

 

참조 : 

https://bae9086.tistory.com/418
https://dataonair.or.kr/db-tech-reference/d-lounge/technical-data/?mod=document&uid=235913
https://scidb.tistory.com/entry/Full-Table-Scan-의-비밀
https://scidb.tistory.com/entry/Oracle11g-에서-Full-Table-Scan의-성능-향상
https://cafe.naver.com/dbian/1673
https://engineering-skcc.github.io/oracle%20tuning/adaptive-direct-path-load/
오라클 성능 고도화 원리와 해법 1 456p
https://positivemh.tistory.com/150
https://www.dba-oracle.com/t_large_small_table_threshold.htm
https://www.dba-oracle.com/t_very_large_object_threshold.htm
https://avdeo.com/tag/_small_table_threshold/
https://oracle-tech.blogspot.com/2014/04/smalltablethreshold-and-direct-path.html
https://energ.tistory.com/entry/Serial-Direct-Scan-기능-소개