2012년 11월 27일 화요일

안드로이드와 수행시간((Elapsed time) 측정

작성자: 박영기
작성일: 2012-11-28


일반적으로 리눅스에서 수행시간 측정을 위해 사용하는 함수는 gettimeofday()입니다.

안드로이드에서는 수행시간 측정을 위해 SystemClock.elapsedRealtime()를 사용합니다.
System.currentTimeMillis()함수는 측정용으로 사용하면 안되는데, 그 이유는 일반적으로 보여주기위한 용도이기 때문에 정확도가 떨어지기 때문입니다.

----------------------------------------------

한동안 시간 처리가 제대로 된다고 생각했습니다.
그런데 SystemClock.elapsedRealtime()역시 시간측정이 제대로 되지 않는군요.
일단 현재까지는 JNI로 gettimeofday()를 호출하는 함수를 따로 만들어야 되지 않을까 하는 생각이 들구요.
좋은 방법을 찾으면 이 글을 갱신하도록 하겠습니다.

참조


2012년 11월 26일 월요일

안드로이드와 이클립스(eclipse) 그리고 AndroidManifest.xml

작성자: 박영기
작성일: 2012-11-27



안드로이드 샘플코드 중 BluetoothChat을 eclipse로 열었을 때 일어난 일이다. 소스코드에 온통 빨간줄이 생기면서 에러표기되는데 Eclipse > Menu > Project > Clean 을 실행하면 잠시동안은 에러표기되는 문제가 사라진다. 하지만 근본적인 해결책은 되지 않았다.

그래서 근본적인 문제를 확인해 보았는데, AndroidManifest.xml파일 때문이었다. xml 파서가 스마트하지 않아서 충돌이 일어난 것인데 이것을 해결하려면 uses-sdk 태그의 속성인 minSdkVersion 앞에 'android:'를 써줘야한다.

올바르게 수정한 코드는 아래와 같은 모습이다.


이상.
업무참조 바란다.

2012년 11월 22일 목요일

안드로이드와 이클립스(eclipse) 그리고 R.java

작성자: 박영기
작성일: 2012-11-23





소스코드는 정상인데 안드로이드 어플이 죽는 현상을 해결하는 방법과
이클립스에서 R.java를 생성하지 못해서 발생하는 에러를 해결하는 방법을 소개하고자 한다.

가장 먼저 할 일은 위 그림에서와 같이 Menu-Project-Clean를 실행해 보는 것이다. 필자는 이걸 실행한 후 어플이 비정상 종료되지 않고 정상적으로 동작하였다. 하지만 소스코드를 편집하는 중에 멀쩡한 코드에서 에러 알림이 계속 발생한다면 다음 과정도 시도해 보길 바란다.

멀쩡한 코드에 대하여 이클립스의 에러 알림이 계속된다면, 이것은 R.java파일과 관련된 문제일 가능성이 높다. 이것을 해결하는 방법은 2가지가 있다.

첫 번째, 프로젝트 네비게이터의 res 폴더에 대문자로 된 파일이 존재하는지 확인한다. 있다면 소문자로 바꿔준다.

두 번째, 아래 그림에서와 같이 Project Property - Java Build Path - Order and Export Tab 에서  Android 버전의 순서를 최상위로 옮겨준다.

여기까지 진행했다면 에러는 더이상 발생하지 않을 것이다.




참조

블루투스와 라이브러리

안드로이드 장비간 블루투스 통신에서 약간의 문제를 확인했다.
페어링을 할 때 팝업으로 버튼을 눌러줘야하는데
입력장치가 없는 장비는 팝업을 눌러줄 수 없다.

한쪽은 임베디드 리눅스를 쓰는 수 밖에 없을 것 같다.

리눅스에서 블루투스를 사용할 수 있는 라이브러리가 있다.
이름은 BlueZ 인데, ODROID-X의 장치명도 BlueZ인걸 보면 같은 것을 사용했나보다.

아래 링크를 참조하면 더 많은 정보를 얻을 수 있을 것 같다. :)


2012년 11월 20일 화요일

Hello Android - Bluetooth 3

작성자: 박영기
작성일: 2012-11-21

이 문서는 Android 개발자 가이드: Bluetooth를 참조하여 작성하였습니다.


discoverability 활성화 시키기 (Enabling discoverability)



용어가 다소 혼동스러울 수 있어서 쉽게 풀어서 설명하겠다. 예를들어, 안드로이드폰과 블루투스 키보드가 있다고 치자. 안드로이드폰은 블루투스 키보드를 발견(discover)할 수 있다.그리고 블루투스 키보드는 안드로이드폰에게 자신을 노출시킬 수 있는 능력(discoverability)를 갖고있다. 이 능력(discoverability)은 On/Off 가능하다. On 상태일 때를 다른말로  discoverable 모드가 활성화 되어있는 상태라고도 한다. 우리말로 설명하면 약간 애매하므로 지금부터 discover와 discoverability, discoverable을 영어 그대로 사용하겠다.

discoverability를 활성화 시키기 위해서는 startActivityForResult(Intent, int) 함수에 ACTION_REQUEST_DISCOVERABLE 액션 인텐트를 사용한다. 이 메소드는 discoverable 모드를 활성화 하도록 시스템에 요청을 보낸다. 기본적으로 디바이스는 120초 동안 discoverable 될 수 있다. 이 시간은 EXTRA_DISCOVERABLE_DURATION 인텐트 엑스트라를 추가하는 것으로 변경할 수 있다. 이 값의 단위는 초이다. 범위는 1부터 3600까지고, 0이 의미하는 것은 항상 discoverable 상태에 있도록 한다는 의미이다. 0보다 작고 3600보다 큰 값은 자동으로 120초로 설정된다. 300초로 설정하는 예제는 다음과 같다:

Intent discoverableIntent = new
Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
startActivity(discoverableIntent);


//
// ODROID-X에서 discoverable 모드가 동작하지 않는다. OTL
// 따라서 이 글은 문제가 해결되면 계속 작성할 것이다.



2012년 11월 19일 월요일

Hello Android - Bluetooth 2

작성자: 박영기
작성일: 2012-11-20
이 문서는 Android 개발자 가이드: Bluetooth를 참조하여 작성하였습니다.


연결된 디바이스 목록 보기


새 Bluetooth 디바이스를 검색하기 전에
이미 연결되어있는지 확인하는 것이 좋다.

위 코드의 동작을 확인하려면
Android Setting 메뉴에서
새 블루투스 디바이스를 연결해야한다.

필자는 블루투스 키보드를 사용하였는데 결과값으로 "i-rocks Bluetooth Keyboard"와 "DC:2C:26:00:4A:29"가 출력되었다. 각각 디바이스 이름과 MAC Address이다.


새 Bluetooth 디바이스 발견하기



// Create a BroadcastReceiver for ACTION_FOUND
private final BroadcastReceiver mReceiver = new BroadcastReceiver() {
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();
        // When discovery finds a device
        if (BluetoothDevice.ACTION_FOUND.equals(action)) {
            // Get the BluetoothDevice object from the Intent
            BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
            // Add the name and address to an array adapter to show in a ListView
            mArrayAdapter.add(device.getName() + "\n" + device.getAddress());
        }
    }
};
// Register the BroadcastReceiver
IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
registerReceiver(mReceiver, filter); // Don't forget to unregister during onDestroy

이 코드의 실행 결과는 흡사 UDP Multicast Group에 참여해서 자신의 존재를 알리기 위해 메시지를 일정시간동안 보내는 것과 유사하다.

BroadcastReceiver class의 객체의 추상메소드인 onReceive()를 구현한 후
android.content.Context class의 registerReceiver() 메소드에 인자로 전달하면
콜백을 받을 준비가 끝난다.

이후 BluetoothAdapter.startDiscover() 함수를 호출하면
약 12초 동안 onRecive() 콜백이 여러번 호출된다.

이건 흡사 Android 기기 > Setting 메뉴 > Bluetooth 에 나오는
Bluetooth 디바이스 목록 페이지와 유사하다.
단, Setting 메뉴에서는 시간 제한이 없다는 차이점이 있다.

주의: 새 Bluetooth 디바이스를 발견하는 일은 꽤 무거운 작업이고, 리소스를 많이 소비한다. 따라서 연결할 디바이스를 찾았다면 확실히 하기 위해서 cancelDiscovery() 를 호출해서 discovery 작업을 중지시켜야한다. 또한 이미 디바이스가 연결된 상태에서 discovery 작업을 할 경우 사용 가능한 대역폭이 아주 많이 감소하게 되므로 주의해야한다.

Hello Android - Bluetooth 1

작성자: 박영기
작성일: 2012-11-20

이 문서는 Android 개발자 가이드: Bluetooth를 참조하여 작성하였습니다.


Bluetooth 권한

블루투스를 사용하는 앱은 다음 두 권한이 필요하다.


manifast 파일을 열어 다음과 같이 권한을 추가한다.




    
    
    

    
        
            
                
                
            
        
    




이들 권한에 대한 자세한 사항은 이 링크를 참고한다.


BluetoothAdapter 객체 얻기


BluetoothAdapter는 전 안드로이드 기기를 통틀어 하나밖에 없다.
다시 말하면, 싱글톤 객체다.
따라서 new를 사용하지 않고 static 메소드로 객체를 얻는다.



BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
if (mBluetoothAdapter == null) {
    // Device does not support Bluetooth
}

Bluetooth 활성화 시키기


Bluetooth가 존재하지만 비활성화 상태일 때에는 이렇게 활성화시킨다.

단, REQUEST_ENABLE_BT는 지역변수이고 0보다 큰 값을 가진다.

if (!mBluetoothAdapter.isEnabled()) {
    Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
    startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT);
}


이 코드를 실행시키면 다음 그림을 볼 수 있다.





앞에서 호출한 startActivityForResult()는 비동기 함수이며 즉시 리턴한다.
수행결과는 잠시후 onActivityResult() 콜백으로 전달된다.

void android.app.Activity.startActivityForResult(Intent intent, int requestCode)
void android.app.Activity.onActivityResult(int requestCode, int resultCode, Intent data)

startActivityForResult()에서 두 번째 인자와 onActivityResult()의 첫 번째 인자가 같다는 것을 눈여겨 보길 바란다.

안드로이드와 버튼(android.widget.Button)

작성자: 박영기
작성일: 2012-11-19

업무상 Android를 해야 할 일이 생겼다.
새로운 플랫폼에서 제일 먼저 해야 할 일은
Hello World를 찍는 일.

익명 메소드를 사용한 예제:

package com.example.helloworld;

import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.widget.Button;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        Button bbb = (Button) findViewById(R.id.button1);
        bbb.setOnClickListener(new Button.OnClickListener() { 
         public void onClick(View v)
         {
          Button btn = (Button) findViewById(R.id.button1);
          btn.setText("Hello Android");
         }
        });
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }
    
}



이번엔 일반적인 예제:

package com.example.helloworld;

import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.widget.Button;

public class MainActivity extends Activity implements View.OnClickListener {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initialize();
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }

    private void initialize() {
        // TODO Auto-generated method stub
        Button btn = (Button) findViewById(R.id.button1);
        btn.setOnClickListener(this);
    }

    public void onClick(View v)
    {
        switch(v.getId())
        {
            case R.id.button1:
                onButton1_click();
                break;
            }
    }

    private void onButton1_click() {
        // TODO Auto-generated method stub
        Button btn = (Button) findViewById(R.id.button1);
        btn.setText("Hello");
    }
}

신텍스 하이라이터(SyntaxHighlighter) 관련

작성자: 박영기
작성일: 2012-11-19

소프트웨어 관련 블로그에서 가장 일반적으로 사용하는 문법 강조 툴은 SyntaxHighlighter이다.

가끔이지만 아래 링크를 참조할 때가 있어서 남긴다.


참조



설치법: http://alexgorbatchev.com/SyntaxHighlighter/integration.html
브러시: http://alexgorbatchev.com/SyntaxHighlighter/manual/brushes/
데모: http://alexgorbatchev.com/SyntaxHighlighter/manual/demo/


2012년 11월 15일 목요일

Android SDK 설치방법 변경

작성자: 박영기
작성일: 2012-11-16


몇 달 전에는 eclipse와 adt-plugin 그리고 SDK를 따로 받아서 설치했어야 했다.
오늘 다시 Android 사이트를 방문해 보니
그 모든 것들이 하나로 통합되어있다.

그냥 압축만 풀고 eclipse를 실행시키면 된다.
That's it!

참조


Android 개발하기 - 목차



Android 블루투스(bluetooth) 개발하기

작성자: 2012-11-16
작성일: 박영기

이 글은 작성중입니다.

참조


저가 ARM 보드 목록

작성자: 박영기
작성일: 2012-11-16

저가 ARM 보드 목록을 보기좋게 정리한 사이트
가운데쯤 ODROID-X 보드가 있다.



Exynos4412 Vs Tegra 3 벤치마크

작성자: 박영기
작성일: 2012-11-16


ODROID-X에 올라온 글을 인용한다.
참조란의 링크를 클릭하면 내용을 볼 수 있다.

Phoronix라는 좀 유명한 리눅스 관련 사이트에 올라온 글입니다.
MIT 대학에서 ODROID-X에 Kernel 3.6과 Ubuntu를 올려서 벤치마킹을 하였습니다.

20여종의 다양한 벤치마킹이 수행되었으며, OMAP4나 TEGRA3에 비해 모든 분야에서 우월한 결과를 보여줍니다.
6페이지 거쳐 자세한 리뷰와 비교가 있으니 참고해 보세요.
Phoronix 사이트가 요즘 많이 느려졌으므로 인내심을 갖고 봐야 합니다.


참조

2012년 11월 13일 화요일

우분투 wubi 디스크 확장하기


wubi 는 우분투(Ubuntu)를 윈도우즈에 쉽게 설치해주는 인스톨러 되시겠다.
그런데 애석하게도 기본 용량이 최대 30GB 이다.

마치며...


참조

https://wiki.ubuntu.com/WubiGuide#How_do_I_create_a_virtual_disk_in_Ubuntu.3F

2012년 11월 11일 일요일

USB 드라이버를 작성하는 방법

작성자: 박영기
작성일: 2012-11-07


나는 리눅스에서 USB 드라이버를 작성하는 방법을 찾고있다.
USB 드라이버를 작성하는 방법은 2가지 정도가 있는 것 같다.

하나는 device driver를 직접 작성하는 것.

다른 하나는 libusb를 사용하는 것.
오피셜 홈페이지는 http://www.libusb.org/ 이다.


참조

파일 디스크립터의 추상적 감시 기능을 제공하는 libevent

작성자: 박영기
작성일: 2012-11-12

select()와 poll()은 이식성이 좋지만 시그널 기반 I/O와 epoll의 성능이 좋으므로 경우에 따라서는 응용 프로그램에서 파일 디스크립터 이벤트를 감시하는 추상 소프트웨어 계층을 만들 필요가 있다.

인용: 리눅스 API의 모든 것. 26장 대체 I/O 모델


참조


2012년 11월 8일 목요일

Makefile 예제

작성자: 박영기
작성일: 2012-11-09

주의: 띄어쓰기 4칸을 TAB으로 바꾸어야 실행 가능함.

응용프로그램용:
CC=g++
CFLAGS=-c -Wall
LDFLAGS=
SOURCES=main.cpp
OBJECTS=$(SOURCES:.cpp=.o)
EXECUTABLE=a.out

all: $(SOURCES) $(EXECUTABLE)

$(EXECUTABLE): $(OBJECTS)
    $(CC) $(LDFLAGS) $(OBJECTS) -o $@

.cpp.o:
    $(CC) $(CFLAGS) $< -o $@

clean:
    rm -f $(OBJECTS)


정적 라이브러리용:
CC=g++
CFLAGS=-c -Wall
SOURCES=name.cpp
OBJECTS=$(SOURCES:.cpp=.o)
TARGET=libXxx

all: $(SOURCES) $(TARGET)

$(TARGET): $(OBJECTS)
    ar cr $(TARGET).a $(OBJECTS)

.cpp.o:
    $(CC) $(CFLAGS) $< -o $@

clean:
    rm -f $(OBJECTS) $(TARGET).a


요건 작업중인거:
#This is prefix for the cross compiler
#Example: arm-linux-
CROSS_COMPILE ?= arm-linux-

CC=$(CROSS_COMPILE)gcc
CXX=$(CROSS_COMPILE)g++

CFLAGS=-c -Wall
LDFLAGS=
SOURCES=main.cpp
OBJECTS=$(SOURCES:.cpp=.o)
EXECUTABLE=a.out

all: $(SOURCES) $(EXECUTABLE)

$(EXECUTABLE): $(OBJECTS) 
    $(CXX) $(LDFLAGS) $(OBJECTS) -o $@

.cpp.o:
    $(CXX) $(CFLAGS) $< -o $@

clean:
    rm -f $(OBJECTS)



코드조각 - getopt

커맨드라인 프로그램 짤 때
가장 많이 쓰는 코드조각이다.

#include 
#include 

void showUsage(char* progname)
{
    printf("Usage: %s [option]\n", progname);
    printf("\t-f option\n");
}


int main(int argc, char**argv)
{

    if(argc < 2)
    {
        showUsage(argv[0]);
        return 0;
    }

    int opt;
    while( (opt=getopt(argc, argv, "hf:")) != -1)
    {
        switch(opt)
        {
        case 'h':
            showUsage(argv[0]);
            break;
        case 'f':
            printf("option is %s\n", optarg);
            break;
        }
    }


    return 0;
}


코드조각 목차

코드 조각들 모음


2012년 11월 6일 화요일

Android 포팅 동영상 강좌 (링크)



작성자: 박영기
작성일: 2012-11-07

총 12개의 동영상으로 구성된 강좌
반드시 보고나면 피가되고 살이되고 뼈가되니
업무 참조 바람



Kernel 옵션 - CPU 전원관리 방법 변경

작성자: 박영기
작성일: 2012-11-07


요즘 핸드폰 같은 임베디드 장비는
전원소비를 줄이기위해서 사용하지 않은 코어를 꺼놓는다.
잠자는 코어는 CPU 사용율이 높아지면 자동으로 깨어난다.

모든 코어가 살아있을 때 성능을 보고싶다면
아래 옵션을 ondemand에서 performance 모드로 바꿔서 커널을 빌드하면 된다.

make menuconfig 
CPU Power Management  --->
    CPU Frequency scaling  --->
        Default CPUFreq governor (performance)  --->

* 'ondemand' 를 'performance' 로 바꿔서 빌드후 적용

이 방법 외에도 방법이 존재한다.
/sys/devices/system/cpu/cpu(N)/online 파일에 1을 쓰면 해당 코어가 살아나게된다.


echo 1 > /sys/devices/system/cpu/cpu1/online


참고로, 하드커널사에서 판매하는 ODROID-X 보드에서는 아래 명령어가 동작하지 않았다.


템플릿


ssh-keygen -t rsa

ssh-keygen 사용법

작성자: 박영기
작성일: 2012-11-07


나는 이 글을 기억하려고 남긴다.

ssh-keygen 명령은 내가 아주 가끔 쓰는 명령이다.

gitosis서버에 public key를 등록 할 때라던지..
github.com에 public key를 등록 할 때라던지..

ssh-keygen -t rsa



2012년 11월 5일 월요일

Nexus 7 커널 빌드하는법

작성자: 박영기
작성일: 2012-11-06
다운로드: config.gz




Nexus 7의 커널을 빌드하는 방법은 다음과 같다:

make tegra3_android_defconfig
make

한가지 흥미로운 점은, Nexus 7의 루트 파일시스템에 config.gz이파일이 존재한다는 점이다. 압축파일 config.gz안에는 config파일이 들어있는데, 위의 1번라인의 명령을 실행한 후에 자동으로 생성되는 .config파일과 내용이 똑같다.

----------------------------------------------------
이글은 An님이 도움주신 내용을 바탕으로 재작성 되었습니다.

Busybox 크로스 컴파일시 주의할 사항

busybox를 크로스컴파일 할 때에도

커널이나

부트로더처럼

$CROSS_COMPILE 환경변수를 사용한다.

make menuconfig에 크로스 컴파일러의
prefix를 설정하는 데가 있는데

$CROSS_COMPILE 변수가 이보다 우선이므로
항상 아래 변수들이 다르게 설정되지 않았는지 확인해야한다

export CROSS_COMPILE=arm-none-eabi-
export PATH=/path/to/toolchain/bin:$PATH
export ARCH=arm
export SUBARCH=arm

망고100 보드로 놀아보자 (링크 모음)

망고카페에서 읽어볼만한 글들을 찾았당~~ 아이 씐나~

이 모든 것을 PDF로 다운로드

2012년 11월 1일 목요일

Nexus 7

목차


repo sync 에러 메시지


에러메시지:
$ repo sync
Fetching projects: 100% (294/294), done.  

error: Exited sync due to gc errors


드래곤볼을 7개 모아

구글신에게 검색
git 버전을 올려야한단다.
참고로 난 지금 git 1.71 쓴다.

sudo add-apt-repository ppa:git-core/ppa
sudo apt-get update
sudo apt-get install git

우분투 10.04 LTS에서는 위 커맨드로 git --version v1.8.0 로 바뀌었다.
우분투 10.10 에서는 위 커맨드로 git --version 1.7.1로 고정되서 바뀌지 않았다. 주의바람.

버전업 하고나니
repo sync 커맨드는 이상무.