[Git] 다른 브랜치와 합치지는 않되 특정 커밋은 가져오고 싶은 경우


여러 브랜치에서 작업을 하다가 공통된 코드에 변경이 있을 경우, 병합은 하지 않으면서 똑같은 커밋을 여러 브랜치에서 사용하고 싶을 때가 있습니다. 그럴 때에는 체리픽 (cherry-pick) 명령어를 사용하면 됩니다.


// 만약에 브랜치1에 있는 abcd1234라는 SHA-1 ID로 시작하는 커밋을 브랜치2에 적용시키려면


git checkout 브랜치2

git cherry-pick abcd1234


[토치 (Torch)] error: more than one operator "==" matches these operands

 

Lua 기반의 딥 러닝 프레임워크인 토치(Torch)를 빌드하다 보면 error: more than one operator "==" matches these operands 와 같은 에러 메시지가 뜰 때가 있습니다. 이 현상은 CUDA가 9.0 버전에서부터 절반 정확도의 부동 소수점 연산을 지원하면서 생기는 문제입니다. 이럴 때에는 빌드 전에 다음과 같이 환경 변수를 추가로 지정해주면 됩니다.

 

export TORCH_NVCC_FLAGS="-D__CUDA_NO_HALF_OPERATORS__"

 

이 글을 쓸 때에 https://github.com/torch/torch7/issues/1086 를 참조하였습니다.

[자바] 밑줄 (_) 이 들어가는 JNI 함수명


JNI에서는 함수명에 자바의 패키지 경로를 모두 쓰게 되는데, 그 때에 .(점)을 _(밑줄)로 치환하게 됩니다. 따라서 자바 함수명 자체에 밑줄이 있으면 문제가 생깁니다. 이런 경우 자바 함수명에 있는 밑줄은 JNI에서 _1로 치환하면 됩니다. 아래는 간단한 예제입니다.


// 자바

package com.tistory.crmn;

public class Hello {

    static native int get_hello();

}



// JNI

extern "C" JNIEXPORT jint JNICALL

Java_com_tistory_crmn_Hello_get_1hello(JNIEnv* env, jobject thiz) { // get_1hello에 주목하세요

    return (jint) 1;

}


전염병이 하나 있다고 하자. 당연한 이야기이지만 전 국민이 모두 예방접종을 맞으면 아무도 그 병에 걸리지 않는다. 복잡한 일은 전 국민 중 일부가 예방접종을 안 맞을 때에 일어난다.

전 국민 중 딱 한 명만 예방접종을 안 맞는다면 그 사람은 예방접종을 안 맞더라도 병에 걸리지 않는다. 다른 사람들이 다 접종을 받았기 때문에 그 사람에게 병을 옮길 사람이 없어서 그렇다. 해외 여행을 갔다가 병에 걸려 올 수는 있겠지만 그럴 확률은 낮다. 전국에서 두 사람만 예방접종을 안 맞더라도 마찬가지다. 남한 인구 5천만 중에서 자기 빼고 예방접종을 안 맞은 다른 한 사람을 마침 그 사람이 병에 걸렸을 때에 만나서 병을 옮아 올 확률은 0에 가깝다.

이렇게 예방접종을 안 맞아도 병에 안 걸리는 사람이 하나 둘 늘어가면 예방접종은 제약회사가 돈을 벌기 위해 만들어낸 개념이라는 음모론이 돌기 시작한다. "누구네 집 애는 예방접종 안 맞았는데도 건강하게 잘만 살더라" 하는 카더라 통신도 함께다. 곧 그런 사람들이 인터넷에 카페를 만들고 무슨 병에는 숯가루를 탄 물이 좋다는 등의 괴소문을 퍼뜨리기 시작한다. 주위에서 그러지 말라고 말리면 자기들이 진리 때문에 핍박을 받는 줄로 생각한다. 이쯤 되면 종교의 영역이다. 이런 집단에는 숯가루 먹고서 병이 나았다는 사람들이 의외로 많이 보이는데 그건 숯가루 먹고 병이 심해진 사람들이 그 카페를 탈퇴해서 그렇다.

자기들끼리만 병에 걸리면 그러든지 말든지라고 할 수도 있겠지만 문제는 이런 사람들이 많아지면 그 자녀들이 피해를 본다는 점에 있다. 전반적인 공중 보건에 악영향을 끼치는 건 말할 것도 없다. 2005년 부터 필수가 된 수두 예방접종을 아직도 안 맞히고, 수두 걸린 아이를 집으로 초대해서 자기 아이에게 수두를 옮기게 하는 어리석은 부모들이 아직도 있다. 애를 때리는 것만이 아동 학대가 아니다. 이런 게 바로 아동 학대다. 그런 부모들이 지금이라도 정신을 차렸으면 좋겠다. 그래서 어릴 때 수두에 걸렸던 자식들이 커서 대상포진으로 고생할 때에는 적어도 따뜻한 물에 숯가루를 타서 마시면 바로 낫는다는 류의 소리 대신 빨리 병원에 가 보라는 말을 하게 되었으면 좋겠다.

[안드로이드] 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에서 열심히 \? 를 시도해보다가 시행착오를 엄청 했습니다.

[안드로이드] Error:Could not find method jackOptions() for arguments


그래들 파일에서 아래 블록 부분을 지워준 뒤 안드로이드 스튜디오를 껐다가 다시 실행하면 문제가 해결됩니다. 


// 이 부분을 삭제하시면 됩니다.

jackOptions {

    enabled true

}


[안드로이드] No toolchains found in the NDK toolchains folder for ABI with prefix: mips64el-linux-android


안드로이드에서 사용되는 NDK 버전이 올라가면서 MIPS형 CPU에 대한 지원이 중단되었습니다. 그런데 build.gradle에 지정되어있는 gradle 버전이 낮으면 안드로이드 스튜디오가 MIPS에 대한 정보를 계속 찾으려고 하다가 위와 같은 에러를 내게 됩니다. 이럴 때에는 gradle 버전을 3.1.4 나 그 이상으로 설정해준 뒤 안드로이드 스튜디오를 껐다가 다시 실행하면 문제가 해결됩니다.


dependencies {

    classpath 'com.android.tools.build:gradle:3.1.4'

}



내가 어릴 때만 해도 복덕방이라는 말을 자주 들을 수 있었다. 복덕방의 업무는 부동산 매매 중개이지만 보통 복덕방이라고 하면 낡은 가죽 소파에 동네 할아버지들이 둘러 앉으셔서 장기를 두시면서 짜장면을 시켜 드시는 모습이 떠오른다. 복덕방은 그렇게 동네 사람들이 무료할 때 모여서 몇 시간씩 시덥잖은 수다를 떨다가 헤어지곤 하는 그런 곳이었다.

내가 고등학생 때 쯤 부터인가, 공인중개사라는 말이 복덕방을 밀어내기 시작했다. 사실 부동산 매매라는 업무의 본질은 바뀐 것이 없었다. 하지만 공인중개사에는 낡은 가죽 소파와 장기판 대신 회색 합성 수지로 코팅된 합판 책상과 펜티엄 컴퓨터, 육중한 CRT 모니터가 있었다. 누런 색 백열등 대신 눈이 쨍한 형광등이 있었고 짜장면 배달은 더 이상 볼 수 없었다.

복덕방(福德房)은 복(福)과 덕(德)이 있는 방(房)이라는 뜻이다. 어감 때문에 구식처럼 보여서 그렇지 사실 뜻이 참 좋은 단어이다. 반면 공인중개사라는 단어에서는 공인(公認)된 중개사(仲介士)라는 뜻 말고는 찾아볼 수 있는 것이 없다. 말을 배우기 시작한 어린이는 있는 그대로를 말하지만 어른은 비유와 상징을 사용한다. 같은 틀에서 볼 때 나는 복덕방이라는 성숙한 단어가 공인중개사라는 유아기적 단어로 퇴행했다고 생각한다.

조선이 건국되고 나라의 중심이 될 궁궐을 지은 뒤 태조 이성계가 정도전에게 궁궐 이름을 지으라고 했다. 그러자 정도전은 유교 경전 중 하나인 시경의 '군자 만년에 큰 복(景福)을 누리리라' 라는 구절을 인용해서 경복궁이라는 이름을 지었다. 뜻도 좋으면서 동시에 유교를 나라의 기반으로 하겠다는 의미를 잘 보여주는 이름이다. 이처럼 1395년, 지금으로부터 500여년 전의 우리 나라 사람들은 한 나라의 중요한 건물 이름을 붙일 때에 적어도 고전 한 수는 읆을 줄 알았고 비유와 상징을 사용할 줄 알았다.

현재 대한민국의 대통령이 사는 곳의 이름은 청와대이다. 청와대(靑瓦臺)는 푸른(靑) 기와(瓦) 집(臺)이라는 뜻이다. 한자로 써 놓으니까 괜히 뭔가 있어보여서 그렇지, 행인이 길을 가다가 "와 저 집은 지붕이 파랗네" 라고 한 것과 다를 것이 없는 이름이다. 깊은 뜻이나 비유나 상징은 찾아볼 수가 없다. 굳이 엮어보려면 유물론하고나 엮일 수 있으려나.

사람은 성장하면서 어휘가 고급스러워지는데 어째 우리말은 요즘 점점 어려지는 것 같다. 복덕방과 공인중개사, 경복궁과 청와대의 차이가 그 현상을 여실히 보여준다. 언제쯤 500년 전으로 성숙할 수 있을까.

'글쓰기' 카테고리의 다른 글

예방접종  (0) 2018.11.23
천하장사 고양이  (0) 2018.10.30
수학으로 설명해본 기독교 교리 1 - 구원  (2) 2018.08.09
"뭐뭐 한 것 같아요"의 유해성  (6) 2018.06.01
너무 규칙적이지 말자  (0) 2018.05.30

+ Recent posts