기본 콘텐츠로 건너뛰기

Template with #pragma pack bug

일반적으로 C/C++컴파일러는 메모리 퍼포먼스를 위해 구조체 멤버를 새로 순서를 정렬하거나 4바이트 단위로 패딩을 한다. 대부분의 컴파일러는 순서는 바꾸지 않고, 전체크기를 4바이트 단위로 패딩을 하는데, 일반상황에서는 크게 문제를 야기 하지 않는다.

그러나 File이나 Network에서 쓰일 고정길이 레코드 개념으로 쓰다보면 4바이트 단위 패딩은 매우 위험하다. 따라서 컴파일러마다 지시자를 두어 1바이트 단위로 패딩할 수 있게 하였다.

GCC는 __attribute__((__packed__)) 지시어를 이용하지만 GCC에만 적용하는 것이고, 일반적으로 VC에서 사용하던 지시자인 #pragma pack을 사용한다.

사용법은 아래와 같다.

#pragma pack(push, 1)

// ... definition ...

#pragma pack(pop)

또는

#pragma pack(1)

// ... definition ...

#pragma pack()

그런데 이러한 Data structure alignment가 Template과 결합하면 심각한 버그를 만들어낸다.

* 2002년에 보고한 내용인데, 2009년에 수정한 GNU GCC팀의 패기에 박수를 보낸다. (응?)


#include
using namespace std;

#pragma pack(1)
template<typename _T>
struct mystr_t
{
_T otype;
int i;
};
#pragma pack()

#pragma pack(1)
struct inner_t
{
char c[3];
};
#pragma pack()

int
main(int argc, char* argv[])
{
cout << "sizeof(inner_t): " << sizeof(inner_t) << endl;
mystr_t<inner_t> tmp;
cout << "sizeof(tmp): " << sizeof(tmp) << endl;

#pragma pack(1)
mystr_t<char[3]> tmp2;
cout << "sizeof(tmp2): " << sizeof(tmp2) << endl;
#pragma pack()

mystr_t<unsigned char[3]> tmp3;
cout << "sizeof(tmp3): " << sizeof(tmp3) << endl;
}

위 코드는 Template이 실제 코드로 구현(instantiatiion)할 때마다 pragma pack이 적용되어 어떠한 경우에도 "7"을 출력해야 정상이다. 그러나 버그가 있는 GCC는 2번째를 제외하고 모두 "8"을 출력한다.

버그이고, 회피 방법은 앞서 말한 __attribute__((__packed__))를 사용하는 방법이 있지만, 소스코드가 지져분해지고 매우 귀찮다. 아니면 GCC를 4.5.1 이후 버전을 사용하는 방법도 있다.

* 참고로 해당 버그가 GCC외에도 꽤 많은 C++ 컴파일러에 존재하였고, 현재는 유명한 컴파일러는 모두 수정한 상태이다.

댓글

이 블로그의 인기 게시물

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

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

Winget 해시 무시하기

가끔씩 Winget 에서 패키지를 다운로드 했을 때, "설치 관리자 해시가 일치하지 않습니다." 오류가 뜰 때가 있다. 보안 이슈가 있지만, 그냥 무시하고 싶을 때, 아래 순서로 무시해준다. 관리자 권한 winget settings --enable InstallerHashOverride 설치 winget install --ignore-security-hash --id NirSoft.NirCmd