/proc 디렉토리는 실제로는 가상 파일시스템입니다. /proc 디렉토리에 들어 있는 파일들은 현재 실행중인 시스템과 커널 프로세스에 대한 정보및 통계 자료를 반영하고 있습니다.
bash$ cat /proc/devices
Character devices:
1 mem
2 pty
3 ttyp
4 ttyS
5 cua
7 vcs
10 misc
14 sound
29 fb
36 netlink
128 ptm
136 pts
162 raw
254 pcmcia
Block devices:
1 ramdisk
2 fd
3 ide0
9 md
bash$ cat /proc/interrupts
CPU0
0: 84505 XT-PIC timer
1: 3375 XT-PIC keyboard
2: 0 XT-PIC cascade
5: 1 XT-PIC soundblaster
8: 1 XT-PIC rtc
12: 4231 XT-PIC PS/2 Mouse
14: 109373 XT-PIC ide0
NMI: 0
ERR: 0
bash$ cat /proc/partitions
major minor #blocks name rio rmerge rsect ruse wio wmerge wsect wuse running use aveq
3 0 3007872 hda 4472 22260 114520 94240 3551 18703 50384 549710 0 111550 644030
3 1 52416 hda1 27 395 844 960 4 2 14 180 0 800 1140
3 2 1 hda2 0 0 0 0 0 0 0 0 0 0 0
3 4 165280 hda4 10 0 20 210 0 0 0 0 0 210 210
...
bash$ cat /proc/loadavg
0.13 0.42 0.27 2/44 1119
|
쉘 스크립트는 /proc에 있는 파일에서 데이터를 뽑아낼 수도 있습니다. [1]
kernel_version=$( awk '{ print $3 }' /proc/version ) |
CPU=$( awk '/model name/ {print $4}' < /proc/cpuinfo )
if [ $CPU = Pentium ]
then
어떤 명령어 실행
...
else
다른 명령어 실행
...
fi |
/proc 디렉토리에는 별나게도 숫자로 된 디렉토리들이 들어 있습니다. 이 서브 디렉토리들은 현재 실행중인 프로세스 ID를 나타냅니다. 각 서브 디렉토리에는 해당 프로세스에 대한 유용한 정보들을 담고 있는 파일이 들어 있는데, stat과 status 파일은 프로세스에 대한 통계 정보를 담고 있고, cmdline 파일은 해당 프로세스가 실행됐을 때의 명령어 줄 인자를 담고 있고, exe 파일은 실행중인 프로세스의 완전한 경로명을 심볼릭 링크하고 있습니다. 각 서브 디렉토리에는 이런식의 다른 파일들도 더 있지만 쉘 스크립팅의 관점에서는 이 정도면 충분할 겁니다.
예 28-1. 특정 PID와 관련있는 프로세스 찾기
#!/bin/bash
# pid-identifier.sh: PID에 해당하는 프로세스의 완전한 경로명을 찾기.
ARGNO=1 # 필요한 인자 수.
E_WRONGARGS=65
E_BADPID=66
E_NOSUCHPROCESS=67
E_NOPERMISSION=68
PROCFILE=exe
if [ $# -ne $ARGNO ]
then
echo "사용법: `basename $0` PID-number" >&2 # 에러 메세지 >표준에러.
exit $E_WRONGARGS
fi
pidno=$( ps ax | grep $1 | awk '{ print $1 }' | grep $1 )
# "ps" 목록에서 pid(첫번째 필드)를 확인.
# 그 다음에는 이 스크립트가 실행시킨 프로세스가 아닌 실제 프로세스인지 확인.
# 마지막의 "grep $1"이 이런 가능성을 제거해 줍니다.
if [ -z "$pidno" ] # 결과가 아무것도 없다면 길이가 0인 문자열이 됩니다.
then # 주어진 pid로 실행중인 프로세스가 없음.
echo "실행중인 프로세스가 없습니다."
exit $E_NOSUCHPROCESS
fi
# 대신 이렇게 해도 됩니다:
# if ! ps $1 > /dev/null 2>&1
# then # 주어진 pid로 실행중인 프로세스 없음.
# echo "실행중인 프로세스가 없습니다."
# exit $E_NOSUCHPROCESS
# fi
# 이 전체 과정은 "pidof" 명령어로 간단하게 할 수 있습니다.
if [ ! -r "/proc/$1/$PROCFILE" ] # 읽기 퍼미션 확인.
then
echo "$1 프로세스는 실행중이지만,"
echo "/proc/$1/$PROCFILE 을 읽을 수 없습니다."
exit $E_NOPERMISSION # 일반 사용자는 /proc 의 몇몇 파일에 접근할 수 없습니다.
fi
# 바로 앞의 두 테스트들을 이렇게 할 수도 있습니다:
# if ! kill -0 $1 > /dev/null 2>&1 # '0' 은 시그널이 아니고 해당 프로세스에
# 시그널을 보낼 수 있는지를 확인해서
# 실행 여부를 알 수 있습니다.
# then echo "PID 가 존재하지 않거나 소유자가 아닙니다." >&2
# exit $E_BADPID
# fi
exe_file=$( ls -l /proc/$1 | grep "exe" | awk '{ print $11 }' )
# 혹은 exe_file=$( ls -l /proc/$1/exe | awk '{print $11}' )
#
# /proc/pid-number/exe 는 그 실행중인 프로세스의
# 완전한 경로명에 대한 심볼릭 링크입니다.
if [ -e "$exe_file" ] # /proc/pid-number/exe 가 존재한다면...
then # 해당 프로세스가 존재하는 것임.
echo "$1 번 프로세스는 $exe_file 으로 실행됐습니다."
else
echo "실행중인 프로세스가 없습니다."
fi
# 이렇게 공을 들여 작성한 이 스크립트는
# ps ax | grep $1 | awk '{ print $5 }'
# 로 "거의" 대치 가능합니다.
# 하지만 'ps'의 5번째 필드가 실행 파일의 경로이름이 아니라
# 그 프로세스의 argv[0] 이기 때문에 제대로 동작하지 않습니다.
#
# 하지만 다음은 제대로 될 겁니다.
# find /proc/$1/exe -printf '%l\n'
# lsof -aFn -p $1 -d txt | sed -ne 's/^n//p'
# Stephane Chazelas 가 추가 설명을 해 주었습니다.
exit 0 |
예 28-2. 온라인 연결 상태
#!/bin/bash
PROCNAME=pppd # ppp 데몬
PROCFILENAME=status # 찾을 곳
NOTCONNECTED=65
INTERVAL=2 # 매 2 초마다 업데이트
pidno=$( ps ax | grep -v "ps ax" | grep -v grep | grep $PROCNAME | awk '{ print $1 }' )
# 'ppp 데몬'인 'pppd'의 프로세스 번호를 찾는데,
# 그 프로세스를 찾기 위한 프로세스 자신은 제거해 줘야 됩니다.
#
# 하지만, Oleg Philon 이 지적했듯이,
#+ 그냥 간단히 "pidof"를 써서 아주 간단하게 할 수 있습니다.
# pidno=$( pidof $PROCNAME )
#
# 여기서 배울 점:
#+ 명령어들이 너무 복잡해 진다면, 간단한 방법을 찾아 볼 것.
if [ -z "$pidno" ] # pid 가 없다면 그 프로세스는 실행중이 아님.
then
echo "연결중이 아닙니다."
exit $NOTCONNECTED
else
echo "연결중입니다."; echo
fi
while [ true ] # 무한 루프, 이 부분은 좀 더 개선될 수 있습니다.
do
if [ ! -e "/proc/$pidno/$PROCFILENAME" ]
# 프로세스가 실행중이면, "status" 파일도 존재합니다.
then
echo "연결이 끊어졌습니다."
exit $NOTCONNECTED
fi
netstat -s | grep "packets received" # 몇 가지 접속 통계.
netstat -s | grep "packets delivered"
sleep $INTERVAL
echo; echo
done
exit 0
# 이 스크립트는 Control-C 로만 끝낼 수 있습니다.
# 독자들을 위한 연습문제:
# "q"를 눌렀을 때 종료하도록 개선시켜 보세요.
# 사용자가 좀 더 쓰기 좋게(user-friendly) 만들어 보세요. |
| 주의 |
보통, /proc 밑에 있는 파일에 쓰기를 하면, 파일 시스템을 손상 시킬 수도 있고 시스템을 망가트릴 수도 있기 때문에 위험합니다. |
| [1] |