Android Camera HAL設計初步

Android Camera HAL設計初步

使用zc301 USB攝像頭,這個攝像頭返回JPEG圖形留,camera的preview需要進行jpeg解碼(沒做),但是可以直接take jpeg照片。

1、修改你的BoardConfig.mk
USE_CAMERA_STUB := false

將stub設置爲false,在編譯時不會編譯android2.1/frameworks/base/camera/libcameraservice中的
CameraHardwareStub.cpp
CameraHardwareStub.h
FakeCamera.cpp
FakeCamera.h
幾個文件

2、hardware下建立Camera HAL目錄,android2.1/hardware/your board/libcamera
複製以上幾個文件
CameraHardwareStub.cpp
CameraHardwareStub.h
FakeCamera.cpp
FakeCamera.h

可以將其重命名
S3C6410CameraHardware.cpp
UsbCamera.cpp

文件中的Fake和Stub同樣可以替換

3、編寫Android.mk文件
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES:= S3C6410CameraHardware.cpp /
UsbCamera.cpp
LOCAL_C_INCLUDES := /
external/jpeg
LOCAL_SHARED_LIBRARIES:= libutils libbinder libui liblog
LOCAL_STATIC_LIBRARIES:= /
libjpeg
LOCAL_MODULE:= libcamera
include $(BUILD_SHARED_LIBRARY)

其中jpeg庫是爲了將來解碼jpeg使用的,根據這個腳本可編譯出libcamera.so

4、修改FakeCamera.cpp->UsbCamera.cpp
目前只實現了基本功能
該文件可以按照V4L2流程來寫
網上也有現成的patch,使用mmap方式,可惜是1.6的,沒編譯過去,待研究

1)構造函數中進行初始化
UsbCamera::UsbCamera(int width, int height)
: mTmpRgb16Buffer(0)
{
fd = open (DEFAULT_DEVICE, O_RDWR /* required */ | O_NONBLOCK, 0);
LOGE("open /dev/video0 fd is %d",fd);
ioctl (fd, VIDIOC_QUERYCAP, &cap);
CLEAR (fmt);
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
fmt.fmt.pix.width = 640;
fmt.fmt.pix.height = 480;
fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV;
fmt.fmt.pix.field = V4L2_FIELD_INTERLACED;
ioctl (fd, VIDIOC_S_FMT, &fmt);
length = fmt.fmt.pix.bytesperline * fmt.fmt.pix.height;
}

2)獲取一幀數據
void UsbCamera::getNextFrameAsYuv422(uint8_t *buffer)
{
LOGE("read to get a pic from camera!");
for (;;)
{
fd_set fds;
struct timeval tv;
int r;
FD_ZERO (&fds);
FD_SET (fd, &fds);
/* Timeout. */
tv.tv_sec = 3;
tv.tv_usec = 0;
r = select (fd + 1, &fds, NULL, NULL, &tv);
if (-1 == r) {
LOGE ("select"); }
if (0 == r) {
LOGE ("select timeout/n"); }
if (read (fd, buffer, length));
break;
}
}

3)析構函數
UsbCamera::~UsbCamera()
{
delete[] mTmpRgb16Buffer;
close (fd);
}


5、修改CameraHardwareStub.cpp->S3C6410CameraHardware.cpp

1)改成640x480吧
void CameraHardware::initDefaultParameters()
{
CameraParameters p;
p.setPreviewSize(640,480);
p.setPreviewFrameRate(1);
p.setPreviewFormat("yuv422sp");//("yuv422sp");
p.setPictureSize(640, 480);
p.setPictureFormat("jpeg");//("jpeg");
if (setParameters(p) != NO_ERROR) {
LOGE("Failed to set default parameters?!");
}
}

2)拍照部分要改,因爲可以直接獲取壓縮的jpeg圖片
int CameraHardware::pictureThread()
{
UsbCamera* usbCamera = mUsbCamera;
if (mMsgEnabled & CAMERA_MSG_SHUTTER)
mNotifyCb(CAMERA_MSG_SHUTTER, 0, 0, mCallbackCookie);
if (mMsgEnabled & CAMERA_MSG_RAW_IMAGE) {
//FIXME: use a canned YUV image!
// In the meantime just make another fake camera picture.
//int w, h;
//mParameters.getPictureSize(&w, &h);
//sp<MemoryBase> mem = new MemoryBase(mRawHeap, 0, w * 2 * h);
LOGE("CAMERA_MSG_RAW_IMAGE");
//UsbCamera cam(w, h);
//cam.getNextFrameAsYuv422((uint8_t *)mRawHeap->base());
//mDataCb(CAMERA_MSG_RAW_IMAGE, mem, mCallbackCookie);
}
if (mMsgEnabled & CAMERA_MSG_COMPRESSED_IMAGE) {
//sp<MemoryHeapBase> heap = new MemoryHeapBase(20000);
//sp<MemoryBase> mem = new MemoryBase(heap, 0, 20000);
//memcpy(heap->base(), kCannedJpeg, 20000);
LOGE("CAMERA_MSG_COMPRESSED_IMAGE");
int w, h;
mParameters.getPictureSize(&w, &h);
sp<MemoryBase> mem = new MemoryBase(mRawHeap, 0, w * 2 * h);
//UsbCamera cam(w, h);
usbCamera->getNextFrameAsYuv422((uint8_t *)mRawHeap->base());
mDataCb(CAMERA_MSG_COMPRESSED_IMAGE, mem, mCallbackCookie);

}
return NO_ERROR;
}

3)status_t CameraHardware::setParameters(const CameraParameters& params)
有個地方只讓take 320x240的pic,要註釋掉
/* if (w != 320 && h != 240) {
LOGE("Still picture size must be size of canned JPEG (%dx%d)",
320, 240);
return -1;
}*/

至此Camera HAL已經可以拍照了。

存在問題:

1)需要做jpeg->YUV以實現preview功能,libjpeg沒用過,暫時不做。
2)UsbCamera.cpp要改成V4L2標準流程,現在這種read模式太簡單,效率低。

發佈了1 篇原創文章 · 獲贊 4 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章