[안드로이드] CMakeLists.txt 파일 설명


안드로이드에서 JNI를 통해 C나 C++ 코드를 사용하려면 CMake나 ndk-build중 하나를 사용해야 합니다. 이 글에서는 CMakeList.txt에 사용되는 여러 옵션에 대해 알아보겠습니다.


1. 사용할 CMake의 최소 버전 설정


cmake_minimum_required(VERSION 3.4.1)



2. 안드로이드에서 사용할 라이브러리의 이름과 공유 및 정적 여부, 소스 코드 경로 설정


아래 예제의 경우 라이브러리의 이름은 native-lib이고 공유 라이브러리이며 소스 코드 경로는 src/main/cpp/native-lib.cpp 입니다. 참고로 이렇게 컴파일을 하면 CMake는 컴파일 결과물을 libnative-lib.so 라는 이름으로 저장합니다.


add_library( native-lib

             SHARED

             src/main/cpp/native-lib.cpp )



3. 소스 코드에서 사용하는 헤더 파일이 들어있는 디렉토리 설정


설정해주지 않아도 동기화 이후에는 자동으로 헤더 파일이 인식되기는 합니다만, 설정해주면 컴파일 단계에서부터 연결된 헤더 파일을 볼 수 있습니다.


include_directories(src/main/cpp/include/)



4. 이미 존재하는 안드로이드 NDK 기본 라이브러리 사용하기


find_library()와 target_link_libraries()를 사용하면 안드로이드 NDK에서 기본으로 제공하는 라이브러리를 따로 컴파일할 필요 없이 바로 사용할 수 있습니다. 아래 예제는 NDK에 있는 log라는 라이브러리를 찾아서 log-lib이라는 변수명으로 저장한 뒤에 native-lib에서 불러와서 사용하는 코드입니다. log-lib이 변수명이기 때문에 target_link_libraries() 에서는 ${log-lib} 으로 표기됩니다.


find_library( log-lib

              log )

target_link_libraries( native-lib

                       ${log-lib} )



5. 소스 코드 형태로 존재하는 안드로이드 NDK 기본 라이브러리 사용하기


4번 경우와 유사한데 이 경우는 안드로이드 NDK 기본 라이브러리가 컴파일되지는 않고 소스 코드 형태로 존재한다는 점이 다릅니다. 사용법은 3번과 4번을 합친 형태입니다.


add_library( app-glue

             STATIC

             ${ANDROID_NDK}/sources/android/native_app_glue/android_native_app_glue.c )


target_link_libraries( native-lib

                       app-glue

                       ${log-lib} )



6. 이미 컴파일 된 외부 라이브러리를 불러와서 사용하기 (매우 중요)


현업에서 안드로이드 개발을 하다 보면 다른 사람이 만든 라이브러리를 이미 컴파일 된 상태로 불러와서 사용해야 할 경우가 정말 많습니다. 이 경우는 add_library()와 set_target_properties()를 사용해야 합니다. add_library()에서는 이 라이브러리가 이미 컴파일 된 상태라는 것을 알려주기 위해 IMPORTED를 사용하고, set_target_properties()에서는 컴파일 된 파일의 경로를 지정해줍니다. 여러가지 CPU 구조를 지원하려면 ANDROID_ABI 변수를 사용하면 됩니다. 컴파일 단계에서 헤더 파일 정보를 보려면 include_directories()를 사용하면 되고, 외부 라이브러리를 불러온 뒤에는 target_link_libraries()를 사용해서 원하는 라이브러리에 연동시키면 됩니다.


아래는 libimported-lib.so 파일을 불러와서 imported-lib 이라는 변수에 저장한 뒤에 native-lib에서 imported-lib을 다른 라이브러리와 함께 사용하는 예제입니다. PROPERTIES IMPORTED_LOCATION는 IMPORTED_LOCATION이라는 값을 정해주겠다는 이야기입니다.


add_library( imported-lib

             SHARED

             IMPORTED )


set_target_properties( imported-lib

                       PROPERTIES IMPORTED_LOCATION

                       imported-lib/src/${ANDROID_ABI}/libimported-lib.so )


include_directories( imported-lib/include/ )


target_link_libraries( native-lib imported-lib app-glue ${log-lib} )



이 정도면 안드로이드에서 일반적으로 CMake를 사용하는 데에는 큰 문제가 없을 것입니다.

대학교 기숙사 주위에는 고양이가 많았다. 그 고양이들을 학교 학생들은 학교 이름을 따서 xx캣이라고 불렀다. 고양이들의 주식은 학생들이 먹고 남긴 배달 음식이었다. 기숙사에 사는 학생들은 종종 치킨, 피자, 족발, 두루치기 등의 다양한 음식을 시켜 먹곤 했다. 학생들이 하도 배달 음식을 많이 시켜먹으니 학교에서 아예 기숙사 건물 입구에 그릇 반납용 선반을 설치해 주었는데, 학생들이 그 곳에 배달 음식 그릇을 가져다 놓고 가면 고양이들이 슬금슬금 다가와서 잔반을 먹곤 했다.

어느 해 겨울이었다. 겨울학기 수업은 듣는 학생이 적어서 학교가 휑했다. 학교가 있던 곳은 평소에 눈이 거의 안 오는 지역이었는데 그 날 따라 눈이 엄청나게 왔었다. 쌓인 눈을 밟으며 밤에 기숙사로 돌아가는데 기숙사 입구에 고양이 두 마리가 있는 것이 보였다. 학교 고양이들은 학생들을 보면 멀찌감치서부터 도망가곤 했는데 얘네들은 희한하게도 나를 보고도 가만히 있었다. 기숙사 입구에 다 가서 보니 고양이들이 비어있는 배달음식 퇴식구를 바라보고 있었다. 겨울방학이라 학생들이 다 집에 가고 나니 잔반이 주식인 고양이들이 먹을 게 없어진 것이었다. 게다가 눈이 와서 추워서 그랬는지 그 고양이들은 눈이 쌓이지 않은 기숙사 처마 밑에서 움직이지 않고 있었다. 못 먹어서 그랬는지 학기중과 비교하면 매우 홀쭉해진 모습이었다.

카드키를 찍고 기숙사로 들어가려는데 자꾸 고양이들이 마음에 걸렸다. 그래도 나랑은 상관없는 일이지 하며 계단을 올라가려는데 발이 안 떨어졌다. 고민을 하다가 학교 매점에 가서 천하장사 소시지를 샀다. 매점까지 갔다 오는 데 시간이 꽤 걸렸는데도 고양이들은 기숙사 입구에 그대로 있었다. 천하장사를 까서 조금 잘라서 던져줬더니 허겁지겁 먹었다. 한 조각 또 던져줬더니 여전히 잘 먹었다. 이번에는 천하장사를 길게 까서 손에 잡고 내밀어 봤더니 가까이 와서 순식간에 다 먹어버렸다. 원래는 고양이에게 천하장사를 하나만 주고 나머지는 내가 먹으려고 했는데 어쩔 수 없이 더 줘야 했다. 두 개째가 지나고 세 개째가 되자 배가 불렀는지 잘 안 받아먹길래 나중에 먹으라고 남은 천하장사를 다 던져주고 방으로 올라갔다. 지금까지 살면서 처음이자 마지막으로 야생 고양이에게 음식을 줘 본 경험이다.

주인 없는 야생 고양이를 일컫는 표준어는 도둑고양이다. 애묘인들 중에는 도둑고양이 대신 길고양이라는 말을 써야 한다고 주장하는 사람들도 있지만 나는 도둑고양이라는 말이 싫지 않다. 꿈보다 해몽이라고, 고양이들은 귀여워서 사람들의 마음을 훔쳐가기 때문에 도둑고양이라고 해석하면 되지 않을까. 그리고 따지고 보면 이 녀석들은 내가 먹으려던 천하장사까지 자기들이 다 먹어버렸으니 도둑이 맞다.

옛날 사진을 뒤져보니 그 때 찍은 사진이 아직 남아 있다. 흰 색 바탕에 검은 색 얼룩 무늬가 있던 고양이. 고양이들 평균 수명을 생각해보면 아마 지금은 이 세상에 없을 거다. 인터넷에 보면 고양이한테 잘 해 줬더니 고양이가 쥐나 벌레를 선물로 잡아왔다는 이야기도 많던데 저 고양이들은 내 천하장사를 다 먹어 놓고는 고맙다는 말 한 마디 없이 입을 싹 닦아 버렸다. 그래도 이렇게 10년 가까이 지난 지금 웃으며 옛날을 추억하게 해 줬으니 이 녀석들은 이미 나한테 천하장사보다 훨씬 큰 보답을 해 준 셈이다.

[VI, VIM] 정규식에서 non-greedy 하게 검색하기 (욕심부리지 않기)


VI의 정규식이 펄이나 파이썬 등의 정규식과 크게 다른 점 중 하나는 non-greedy (욕심부리지 않기) 연산자입니다. 대부분의 경우 물음표 기호가 사용되는데 VI에서는 \{-}가 사용됩니다. 예를 들면 다음과 같습니다.


:%s/hello.\{-}://g


라고 하면 hello에서 가장 가까이 있는 : 까지만 사라지게 됩니다. 펄이나 파이썬이었으면 .? 이런 식으로 사용했을 거라, 지레짐작으로 VI에서 열심히 \? 를 시도해보다가 시행착오를 엄청 했습니다.

+ Recent posts