리눅스
OOM Killer와 oom_score에 대해 알아보자
겅겅겅
2022. 12. 23. 10:12
OOM Killer (Out of Memory Killer)란?
메모리가 부족할 경우 운영체제의 설정에 따라 특정 어플리 케이션을 강제로 종료하는 기능입니다.
OOM Killer가 동작에 대해서는 /var/log/syslog에 로그가 남게 됩니다.
oom_score란?
OOM_Killer에서 가지는 종료 기준값으로 가장 높은 oom_score를 먼저 종료시킵니다.
score는 0 ~ 1000까지의 범위를 가지게 됩니다.
/proc/<pid>/oom_score 명령어를 통해 해당 프로세스의 oom_score를 조회 할 수 있습니다.
기본적으로 리눅스에서 1번 init프로세스는 0 의 oom_score값을 가집니다. (저는 wsl을 통해 조회해서 666이 나왔습니다.)
oom_score의 선정기준
oom_score를 높게 주는 기준은 다음과 같습니다.
- 완료된 작업의 수를 최대로 유지할 수 있어야한다.
- 많은 양의 메모리를 복수 할 수 있어야한다.
- 메모리 소비량이 적은 프로세스는 죽이지 않는다.
- 최소한의 프로세스만 종료시킨다.
- 사용자가 원하는 프로세스를 종료시킨다.
소스코드 (v5.19)
oom_kill.c 파일의 oom_badness() 메소드에서 oom_score를 계산하게됩니다.
long oom_badness(struct task_struct *p, unsigned long totalpages)
{
/*
* points가 최종 리턴할 oom_score값입니다.
*/
long points;
long adj;
if (oom_unkillable_task(p))
return LONG_MIN;
p = find_lock_task_mm(p);
if (!p)
return LONG_MIN;
/*
* adj값을 최소로 하려면
* oom_score_adj값을 최소값인 OOM_SCORE_ADJ_MIN 즉 -1000으로 설정하면 됩니다.
*/
adj = (long)p->signal->oom_score_adj;
if (adj == OOM_SCORE_ADJ_MIN ||
test_bit(MMF_OOM_SKIP, &p->mm->flags) ||
in_vfork(p)) {
task_unlock(p);
return LONG_MIN;
}
/*
* rss값과 pagetable의 크기가 크면 높은 point를 가지게 됩니다.
*/
points = get_mm_rss(p->mm) + get_mm_counter(p->mm, MM_SWAPENTS) +
mm_pgtables_bytes(p->mm) / PAGE_SIZE;
task_unlock(p);
/*
* 최종 oom_score값은 point에 adj값을 더한 값으로 결정됩니다.
*/
adj *= totalpages / 1000;
points += adj;
return points;
}
아래 코드와 같이 oom_score는 adj와 points의 값으로 결정됩니다.
우리는 리눅스에서 adj값을 설정하여 프로세스의 oom_score값을 변경 할 수 있습니다.
oom_score 변경 명령어
echo adj값 > /proc/<pid>/oom_score_adj 명령어롤 통해 oom_scoring을 할 수 있습니다.
- 1000을 주게되면 oom_score값이 0이 됩니다.