2012년 6월 13일 수요일

허니컴 이전의(그리고 하드웨어 보강 이전의) bitmap의 out of memory 관련 해결법 recycle()


Bitmap size exceeds VM budget 에러 메세지때문에 빡차서 쓰는 글이다.

** 결론부터 말하자면,
Android Activity 의 lifecycle 과, Object lifecycle 은 별개로 동작하기 때문에 발생하는 문제라 볼 수 있다.
VM 의 GC는 Object 가 죽은 상태(멤버의 참조 횟수가 0이고, 자신의 참조 횟수가 0일때)일 때만 수행되는데, Activity 가 죽었다 하여 Object 가 죽은 것은 아니기 때문에 Bitmap 에 대한 참조가 남아 있어 GC를 하지 못하는 것이다.
이는 좀더 유연한 프로그래밍을 가능하게 하는 안드로이드 개발팀의 배려라 볼 수도 있겠지만, 어차피 Activity 코딩시에는 Activity 생명주기를 따라 코딩하는게 일반적이니 Android framework 에서 이정도는 강제로 해 주는게 어떨까 싶다. 아니면 Manifest 에 옵션을 넣어 주던가.


아래와 같은 간단한 Activity 를 실행해 보자. 단 22라인의 이미지 파일의 크기는 3888x2592 (1000만화소) 이며 정확히 20155392 바이트 (19.22MB) 의 메모리를 필요로 한다. 실행 기기는 Nexus S (버전 2.3.2) 로 32Mb 의 가용 메모리(heap)를 가지고 있기 때문에 문제없이 화면이 뜰 것이다.

31라인의 imageView.setImageBitmap(null); 를 주목하자.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
package net.oomtest;
 
import android.app.Activity;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.util.Log;
import android.widget.ImageView;
import android.widget.LinearLayout;
 
public class OomtestActivity extends Activity {
    LinearLayout mainLayout;
    ImageView imageView;
     
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
         
        mainLayout = (LinearLayout) findViewById(R.id.mainLayout);
        imageView = new ImageView(this);
        imageView.setImageBitmap(BitmapFactory.decodeFile(
            "/mnt/sdcard/hwan.documents/images/bigimage/1_portrait.jpg"));
         
        mainLayout.addView(imageView);
    }
     
    @Override
    protected void onDestroy() {
        Log.d("OOMTEST", "onDestroy");
         
        imageView.setImageBitmap(null);
        super.onDestroy();
    }
}

앱을 실행한 결과 화면은 아래와 같다.


아래는 앱 실행 직후 메모리 사용 상황이다. allocated: 부분을 유심히 보길 바란다.
(native 23745kB, dakvik 2763kB, total 26508kB)

Applications Memory Usage (kB):
Uptime: 22510282 Realtime: 28703998

** MEMINFO in pid 8286 [net.oomtest] **
                    native   dalvik    other    total
            size:    23764     5379      N/A    29143
       allocated:    23745     2763      N/A    26508
            free:       18     2616      N/A     2634
           (Pss):      602      137    22165    22904
  (shared dirty):     2208     1848     7276    11332
    (priv dirty):      508       56    20752    21316

 Objects
           Views:        0        ViewRoots:        0
     AppContexts:        0       Activities:        0
          Assets:        2    AssetManagers:        2
   Local Binders:        5    Proxy Binders:       10
Death Recipients:        0
 OpenSSL Sockets:        0

 SQL
               heap:        0         MEMORY_USED:        0
 PAGECACHE_OVERFLOW:        0         MALLOC_SIZE:        0


 Asset Allocations
    zip:/data/app/net.oomtest-1.apk:/resources.arsc: 1K

19MB의 대용량 이미지가 별 탈 없이 뜨는게 놀랍다. 그렇다면 이제 화면을 회전시켜 보도록 한다.
안드로이드의 Activity 는 onPause -> onStop -> onDestroy 의 생명주기를 타며 화면이 새로 그려질 때, onCreate -> onStart -> onResume 의 순으로 Activity 가 다시 시작된다. 화면방향이 바뀌면 현재 화면의 내용을 모두 버리고 새로운 화면을 그리게 되므로 우리의 코드는 설명한 모든 생명주기를 다 타게 된다.

(여담이지만 이로 인해 화면 방향전환이 꽤 느리며 그것을 회피할 수 있는 방법은 http://developer.android.com/guide/topics/resources/runtime-changes.html 의 'Handling Configuration Change Yourself' 섹션 및 http://developer.android.com/resources/articles/faster-screen-orientation-change.html 를 참고하면 된다. 이 글의 주제인 OOM 과는 무관한 내용이므로 생략)

화면 방향을 바꾸면 아마 아래와 같이 Exception 이 발생하며 앱이 죽을 것이다.


01-02 21:52:22.515: D/OOMTEST(9993): onDestroy
01-02 21:52:22.585: D/dalvikvm(9993): GC_EXTERNAL_ALLOC freed 40K, 49% free 2760K/5379K, external 21308K/23356K, paused 63ms
01-02 21:52:22.601: E/dalvikvm-heap(9993): 20155392-byte external allocation too large for this process.
01-02 21:52:22.613: E/GraphicsJNI(9993): VM won't let us allocate 20155392 bytes
01-02 21:52:22.624: D/dalvikvm(9993): GC_FOR_MALLOC freed <1K, 49% free 2760K/5379K, external 1625K/21308K, paused 14ms
01-02 21:52:22.624: D/skia(9993): --- decoder->decode returned false
01-02 21:52:22.628: D/AndroidRuntime(9993): Shutting down VM
01-02 21:52:22.628: W/dalvikvm(9993): threadid=1: thread exiting with uncaught exception (group=0x40015560)
01-02 21:52:22.628: E/AndroidRuntime(9993): FATAL EXCEPTION: main
01-02 21:52:22.628: E/AndroidRuntime(9993): java.lang.OutOfMemoryError: bitmap size exceeds VM budget
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.graphics.BitmapFactory.nativeDecodeStream(Native Method)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:470)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:284)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:309)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at net.oomtest.OomtestActivity.onCreate(OomtestActivity.java:24)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1611)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1663)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:2832)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread.access$1600(ActivityThread.java:117)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:935)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.os.Handler.dispatchMessage(Handler.java:99)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.os.Looper.loop(Looper.java:130)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at android.app.ActivityThread.main(ActivityThread.java:3683)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at java.lang.reflect.Method.invokeNative(Native Method)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at java.lang.reflect.Method.invoke(Method.java:507)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
01-02 21:52:22.628: E/AndroidRuntime(9993):  at dalvik.system.NativeStart.main(Native Method)

역시 예상대로 OOM Error 다. 그런데 왜 화면전환시 OnDestroy 에서 imageView.setImageBitmap(null); 를 호출해 주었는데도 OOM 이 발생할까?

왜냐하면 imageView 에 할당된 imageBitmap 에 대한 참조를 끊어 주었지만 ImageView 가 화면에 그리기 위한 Bitmap 은 VM Heap 에 할당된 것이 아니라 Native heap 에 할당되어 있다. 따라서 이를 null로 끊었다 하더라도 Native 영역에 대해 free()를 하지 않아 할당내용이 dangling pointer 가 되어 그대로 메모리 누수로 이어져 버리기 때문이다. Bitmap 은 내부의 픽셀정보를 Dalvik VM Heap 를 사용하지 않고 Native heap 을 사용하여 저장하고 있으며 따라서 이 부분은 JNI 로 구현되어 있다.

Java 만 프로그래밍 하던 분들께는 다소 낯선 이야기가 될 수도 있지만 이 dangling pointer 문제는 많은 c 개발자들의 골치를 썩이던 문제였으며 이를 해결할 방법은 오직 한가지, malloc 의 끝에서는 반드시 free를 해 주라는 것 뿐이다.

첫 화면에서 사용한 bitmap 은 화면전환 이후 더 이상 필요없으므로, 첫 화면의 onDestroy() 호출 이후에 참조가 제거되어야 함에도 제거되지 않아 VM 이 GC를 제대로 해 주지 못해 발생하는 현상이다. 프레임웍은 지금같은 상황은 화면(Activity)이 바뀌지 않았으므로 이미지를 버릴 필요가 없다고 판단한 모양이다. 허나 첫 화면이 onDestroy 로 사라졌으면 그 화면 내에서 선언한 내용들은 GC 대상이 되어야 함에도 그렇지 못하다.

불만인 점은, 우리는 Java 로 코딩하며 VM 을 사용하고 있음에도 왜 이런 부분까지 프레임웍에서 챙겨주지 못하냐는 점이다. 이에 대한 불만의 글들이 상당히 많고 메모리 릭 관련 글을 쓴 Android 개발팀의 Romain Guy 가 역관광 당해버린 글타래마저 있으니 참고하자.(http://code.google.com/p/android/issues/detail?id=8488)

위 문제를 해결하기 위해선, Bitmap.recycle() 를 호출해 주어야 하며 ImageView 에 사용한 bitmap 을 recycle 하는 코드는 아래와 같다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
package net.oomtest;
 
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.util.Log;
import android.widget.ImageView;
import android.widget.LinearLayout;
 
public class OomtestActivity extends Activity {
    LinearLayout mainLayout;
    ImageView imageView;
     
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
         
        mainLayout = (LinearLayout) findViewById(R.id.mainLayout);
        imageView = new ImageView(this);
        imageView.setImageBitmap(BitmapFactory.decodeFile("/mnt/sdcard/hwan.documents/images/bigimage/1_portrait.jpg"));
         
        mainLayout.addView(imageView);
    }
     
    @Override
    protected void onDestroy() {
        Log.d("OOMTEST", "onDestroy");
        recycleBitmap(imageView);
         
        super.onDestroy();
    }
     
    private static void recycleBitmap(ImageView iv) {
        Drawable d = iv.getDrawable();
        if (d instanceof BitmapDrawable) {
            Bitmap b = ((BitmapDrawable)d).getBitmap();
            b.recycle();
        } // 현재로서는 BitmapDrawable 이외의 drawable 들에 대한 직접적인 메모리 해제는 불가능하다.
         
        d.setCallback(null);
    }
}

코드를 위와 같이 바꾸고 화면을 여러번 돌려보면 OOM Error 없이 화면전환이 잘 되는 것을 확인할 수 있다.
recycle 한줄 호출을 위해 인스턴스 검사, 형변환 까지 해야 하다니 최악이다.


별것 아닌 팁 처럼 보일 것이다. 그러나 왜 이 글을 포스팅하냐면, Android SDK 문서대로 하다가 낭패를 보는 경우가 왕왕 있기 때문이다. Bitmap.recycle() 에 대한 설명을 보도록 하자.

public void recycle ()

Since: API Level 1
Free the native object associated with this bitmap, and clear the reference to the pixel data. This will not free the pixel data synchronously; it simply allows it to be garbage collected if there are no other references. The bitmap is marked as "dead", meaning it will throw an exception if getPixels() or setPixels() is called, and will draw nothing. This operation cannot be reversed, so it should only be called if you are sure there are no further uses for the bitmap. This is an advanced call, and normally need not be called, since the normal GC process will free up this memory when there are no more references to this bitmap.
영어가 골치아픈 분들을 위해 해석해 보자면 아래와 같다.

public void recycle ()

 API Level 1 부터 가능
비트맵에 연관된 네이티브 오브젝트를 정리하고, 픽셀 데이터에 연관된 참조를 끊는다. 이 메소드를 호출한다 하여 픽셀 데이터가 즉시 정리되지는 않는다. 이 비트맵에 대한 다른 참조가 없다면 GC의 대상이 되도록 해 줄 뿐이다. 이 비트맵은 "죽은" 상태가 되어 getPixels() 또는 setPixels() 과 같은 메소드를 호출하면 예외를 발생시킨다. 또한 화면에 아무 것도 그리지 않을 것이다. 이 동작은 되돌릴 수 없으므로 이 비트맵을 더 이상 사용하지 않는다는 확신이 들 경우에만 호출하도록 한다. 이 기능은 고급 기능이며 일반적으로 사용할 필요가 없다. 왜냐하면 일반적인 GC 상황이 되면 비트맵에 대한 참조가 없을 경우 자동으로 GC 대상이 되어 메모리가 해제되기 때문이다.
구라도 이런 구라가 없다.
GC 상황이 되면 메모리가 해제된다고 했는데 구라다.
일반적으로 호출될 필요가 없다고 했는데 안 하면 앱이 죽으므로 구라다.
만약 안드로이드 개발팀이 거짓말을 적어놓은 게 아니라면, 이는 명백한 프레임워크 버그다. (처음 소스에서 drawable 을 null로 설정했기 때문에 onDestroy 이후에는 네이티브니 뭐니 소리가 나오기 전에 GC가 되어야 한다.)

왜냐하면 위의 코드에서 보듯 imageView 와 bitmap 의 사용 scope 는 onCreate 메소드만으로 한정되어 있기 때문에, 라이프사이클 동안 자동으로 정리되어야 함에도 실상은 그렇지 않기 때문이다. 우리가 반드시 별도의 recycle 과정을 추가로 해 주어야 문제가 사라진다. 또한 메모리 릭 피하기(http://blog.naver.com/nimbusob/147042528) 란 글에서 보면 drawable 을 View 에 주입하면 내부적으로 callback 이 등록되어 어쩌구 저쩌구가 일어나고 이게 Activity context 를 참조하니 마느니 하는 소리를 하는데 개발자가 이런 내부 구현까지 알아야 된다는것 자체가 문제가 있다또한 이 문제에 대한 전 세계 개발자들의 불만에 대해 Android team 은 아직까지도 침묵으로 일관하고 있다.

좀더 구체적으로 이야기 하자면, 사실 위 문제는 정확한 GC 타이밍을 잡지 못해 발생한 문제인데 왜 그런지에 대한 추측을 해보자면 다음과 같다. 결국 Native heap과 VM heap 은 분리되어 있으며 GC는 VM heap 이 꽉 찼을때 일어나는 것이므로 위의 상황(Native heap 23745kB/23764kB, VM heap 2763kB/5379kB) 에서는 VM heap 은 아직 여유가 있으므로 GC를 하지 않아 결국 OOM 이 발생한 것이다. (실상은 Native heap 이 넘치기 직전이라 GC를 해야 함에도 불구하고) 아마 Dalvik VM 의 가비지 컬렉터는 Native heap 을 모니터링하지 않는 것(못하는 것)이 아닐까.

이 문제와 관련하여 더욱 자세한 안드로이드 비트맵과 메모리에 관한 토론글은 (http://stackoverflow.com/questions/1945142/bitmaps-in-android) 에서 확인할 수 있다.

따라서 위와같은 예제 뿐 아니라, xml 레이아웃에서 설정한 Drawble 들도 BitmapDrawable 이라면 명시적으로 recycle 을 해 주는 것이 속편하다. 언제 메모리가 확보될 지 알 수 없는 상황에서 OOM으로 우리 앱이 뻗어버리는 것 보단 나을테니.

이때까지 안드로이드의 프레임웍 자체의 문제를 확인해 보았다. 2.2 이상에서는 해결되었다는 Romain Guy 의 변명글이 있긴 하지만 개발자 커뮤니티에서는  그것조차도 구라라고 하니(테스트환경 2.3.2 에서도 여전히 고쳐지지 않았다) Bitmap 사용에 관해서만큼은 버그이던 뭐던, 시스템을 믿지 말고 구식(old school) 스타일로 우리가 직접 객체 주기를 관리하는 편이 낫다.

또한 이와 비슷한 문제로 View 들의 callback 및 리소스 할당해제에 대한 지저분한 트릭이 있는데 다음 포스트에서 이 문제에 대해 다뤄보겠다.



댓글 없음:

댓글 쓰기