이것이 점프 투 공작소

OOM Killer와 oom_score에 대해 알아보자 본문

리눅스

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이 됩니다.