기본 콘텐츠로 건너뛰기

gdb로 errno를 붙잡아보자

가끔씩 Linux에서 GDB로 errno를 보고 싶을 때가 있는데, 그럴 때마다 "Cannot access memory at address 0x8"를 뱉어내곤 한다. 쩝쩝 errno가 멀티 스레드 환경에서 여기저기에서 세팅되는 현상을 막기 위해 함수(스레드에 따라 다른 errno 주소를 반환)로 구현한 것을 매크로로 재설정했기 때문이다.

매크로는 /usr/include/bits/errno.h에서 아래와 같이 정의하였다.
#define errno (*__errno_location ())


만약 errno 값을 보고 싶으면 일단 프로세스를 돌리고
(gdb) p (*__errno_location())


watch는 좀 복잡하다. 일단 errno의 포인터를 얻고 그것이 가르키고 있는 곳에 값을 감시하도록 하자.
(gdb) p/x (__errno_location())
$1 = 0xb7fea6a0
(gdb) watch (*(int*)0xb7fea6a0)
Hardware watchpoint 2: *(int *) 3086919328
(gdb) c
Continuing.
Hardware watchpoint 2: *(int *) 3086919328

Old value = 0
New value = 22

0x0050c885 in __read_nocancel () from /lib/tls/libc.so.6


뭐 이런 것이다. 근데 왜 errno를 __thread int로 정의하지 않은걸까?

댓글

이 블로그의 인기 게시물

Bash Array, Map 정리

Bash에서 Array, Map에 대한 정리. (매번 찾기 귀찮) 찾아보진 않았지만, Bash에서 Array든 Map이든 동일하게 Map(C++에서 Unordered Map)으로 동작하는 것 같다. 왜냐하면, Array의 Index가 연속하지 않아도 동작한다. 그저 Key가 0 이상의 정수인 Map이랑 비슷하게 동작한다. 예) 1, 2, 3, 9, 10 Array # 생성 declare -a empty_array declare -a ar=(haha hoho baba "long string haha hoho") # 접근 echo "ar[0]=${ar[0]}" echo "all as array=${ar[@]}" # 큰따옴표 안에서 각 원소를 따로따로 전달한다. echo "all as one=${ar[*]}" # 큰따옴표 안에서 각 원소를 문자열 하나로 합쳐 전달한다. echo "indexes=${!ar[@]}" echo "indexes=${!ar[*]}" echo "length=${#ar[@]}" echo "length=${#ar[*]}" echo "last=${ar[-1]}" echo "last=${ar[@]: -1}" # 콜론 뒤에 빈 칸이 꼭 필요하다. 옛 방식 # 현재 상황 declare -p ar #(출력) declare -a ar=([0]="haha" [1]="hoho" [2]="baba" [3]="long string haha hoho") ar[100]=hello # 인덱스를 건너 뛰어도 동작한다. declare -p ar #(출력) declare -a ar=([0]="haha" [1]="hoho" [2]="baba" [3]=&

설치한 패키지에서 RPM 추출하기

오래된 패키지를 관리할 저장소가 없어졌고, 기존 패키지로 다른 서버를 세팅해야할 일이 생겼다면 RPM의 리패키지 기능을 이용해보자. $ rpm -e --repackage [PACKAGE_NAME] 위와 같이 리패키지하면, /var/spool/repackage/ 에 생성한 RPM파일이 있다. :-)