솔라리스에서 dirent 구조체의 d_name 멤버가 문제를 일으킵니다~~
글쓴이: superkkt / 작성시간: 금, 2006/02/17 - 1:14오전
int read_dir(struct dirent **ptr, char *target, size_t *size) { DIR *dir; struct dirent *dp; int cnt = 0; if(*ptr == NULL) { return -1; } if((dir = opendir((target == NULL) ? "." : target)) == NULL) { RETURN(E_OPENDIR); } while((dp = readdir(dir)) != NULL) { (*ptr)[cnt] = *dp; if(cnt++ == *size - 1) { *size += *size; if((*ptr = realloc(*ptr, *size)) == NULL) { return -1; } } } return cnt; }
ftp server에서 디렉토리에 있는 파일을 읽어서 동적으로 할당된 struct dirent 배열에 저장을 합니다. 그리고나서 알파벳순으로 정렬하는 함수에 이 배열을 던져주는데요..
리눅스에서는 제대로 되는데 솔라리스에서만 정상작동을 안해서 디버깅을 해봤더니 이런 결과가 나옵니다.
while((dp = readdir(dir)) != NULL) { (*ptr)[cnt] = *dp;
이 부분에서 dp->d_name을 확인해보면 파일명이 제대로 들어가 있는데 (*ptr)[cnt]->d_name을 확인해보면 파일명이 맨앞 한글자씩만 들어가고 다음 자리부터는 쓰래기 값이 들어가있습니다.
그래서 /usr/include/sys/dirent.h를 확인해보니 dirent 구조체가 아래와 같이 선언이 되어있더군요.
/* * File-system independent directory entry. */ typedef struct dirent { ino_t d_ino; /* "inode number" of entry */ off_t d_off; /* offset of disk directory entry */ unsigned short d_reclen; /* length of this record */ char d_name[1]; /* name of file */ } dirent_t;
d_name이 1바이트 크기의 배열인데 저것 때문에 파일이름 맨 앞 한글자만 저장이 되는건지는 모르겠습니다. 아무튼 궁금한건 저게 포인터도 아닌데 어떻게 dp->d_name을 string으로 출력해보면 완전한 파일 이름이 보이는지 모르겠네요.
일단은 dirent 구조체 전체를 복사할 필요가 없을것 같아서 파일네임만 뽑아서 복사하는걸로 고치는 중입니다.
그런데 무지 궁금하네요.. d_name[1] 이게 어떻게 작동을 하는지..
Forums:
[url=http://www.cinsk.org/~cinsk/cfaqs/h
C FAQ 2.6 참고하세요.
그리고, 문제가 되는 코드는 바로
입니다.
C FAQ: http://www.eskimo.com/~scs/C-faq/top.html
Korean Ver: http://cinsk.github.io/cfaqs/
[quote="cinsk"][url=http://www.cinsk.org
malloc(sizeof(struct dirent))로 할당된 메모리에 malloc(sizeof(struct dirent) - 1 + strlen(filename) + 1))로 할당된 메모리를 대입하려고해서 생기는 문제라고 생각되는데.. 제가 제대로 본건가요?? :oops:
그럼 해결책은 char filename[PATH_MAX]를 만들어서 여기다 dp->d_name만 복사하는게 제일 좋겠네요.
항상 좋은 답변 주시는 cinsk님 정말 감사합니다. 참 신기한게 매번 질문할때마다 C FAQ에 답이 있네요.. 에허.. 제가 검색도 제대로 안해보고 질문부터 덥썩 올렸다는 얘기겠죠~ 앞으론 막히는게 있으면 C FAQ부터 다시 한번 읽어봐야 겠습니다. 8)
======================
BLOG : http://superkkt.com
댓글 달기