publicstatic Camera mCameraDevice;java
try {android
Log.d(TAG,"open camera 0");api
//writeFile(APK_FILE_PATH,1);app
Log.d("qyh","opencamera"+System.currentTimeMillis());函數
mCameraDevice = Camera.open(0);/*打開camera,獲取camera實例給mCameraDevice */oop
Log.d("qyh","sendMessage"+System.currentTimeMillis());學習
mHandler.sendEmptyMessageDelayed(DISMISS_ALL,500);ui
}catch (Exception e) {this
Log.e(TAG,"open camera error");google
//writeFile(APK_FILE_PATH,0);
e.printStackTrace();
return;
}
try{
mCameraDevice.setPreviewDisplay(holder);/*這個應該是數據的顯示部分,待分析*/
}catch (Exception e) {
e.printStackTrace();
}
Camera.Parameters parameters = mCameraDevice.getParameters();/*獲取驅動的相關參數*/
parameters.setPreviewSize(800, 600);/*從新設置預覽的分辨率*/
mCameraDevice.setParameters(parameters);/*設置相關參數*/
mCameraDevice.startPreview();/*開始進入預覽模式*/
以上是應用APP中打開camera的簡要步驟,下面咱們就對mCameraDevice = Camera.open(0);進行具體的跟進下:
STEP 1:
\frameworks\base\core\java\android\hardware\Camera.java
public static Camera open(int cameraId) {
if (!isPermissionGranted()) {
return null;
}
return newCamera(cameraId);
}
STEP 2:
\frameworks\base\core\java\android\hardware\Camera.java
/** used by Camera#open, Camera#open(int)*/
Camera(int cameraId) {
int err = cameraInitNormal(cameraId);
if (checkInitErrors(err)) {
switch(err) {
case EACCESS:
throw newRuntimeException("Fail to connect to camera service");
case ENODEV:
throw newRuntimeException("Camera initialization failed");
default:
// Should never hit this.
throw newRuntimeException("Unknown camera error");
}
}
}
STEP 3:
private int cameraInitNormal(int cameraId){
return cameraInitVersion(cameraId,CAMERA_HAL_API_VERSION_NORMAL_CONNECT);
}
/*
CAMERA_HAL_API_VERSION_NORMAL_CONNECT參數的定義
private static final int CAMERA_HAL_API_VERSION_NORMAL_CONNECT= -2;
*/
STEP 4:
private int cameraInitVersion(int cameraId,int halVersion) {
mShutterCallback = null;
mRawImageCallback = null;
mJpegCallback = null;
mPreviewCallback = null;
mPreviewRawDumpCallback = null;
mPostviewCallback = null;
mUsingPreviewAllocation = false;
mZoomListener = null;
Looper looper;
if ((looper = Looper.myLooper()) !=null) {
mEventHandler = new EventHandler(this,looper);
} else if ((looper =Looper.getMainLooper()) != null) {
mEventHandler = newEventHandler(this, looper);
} else {
mEventHandler = null;
}
return native_setup(newWeakReference<Camera>(this), cameraId, halVersion,
ActivityThread.currentOpPackageName());
}
/*
private native final int native_setup(Object camera_this, int cameraId,int halVersion,
String packageName);
native_setup是一個本地的方法,對應的JNI在下面步驟中。
*/
STEP 5:
\frameworks\base\core\jni\android_hardware_Camera.cpp
// connect tocamera service
static jint android_hardware_Camera_native_setup(JNIEnv *env,jobject thiz,
jobject weak_this, jint cameraId, jinthalVersion, jstring clientPackageName)
{
// Convert jstring to String16
const char16_t *rawClientName =reinterpret_cast<const char16_t*>(
env->GetStringChars(clientPackageName, NULL));
jsize rawClientNameLen =env->GetStringLength(clientPackageName);
String16 clientName(rawClientName,rawClientNameLen);
env->ReleaseStringChars(clientPackageName,
reinterpret_cast<const jchar*>(rawClientName));
sp<Camera> camera;
if (halVersion ==CAMERA_HAL_API_VERSION_NORMAL_CONNECT) {
// Default path: hal version is don'tcare, do normal camera connect.
camera = Camera::connect(cameraId,clientName,
Camera::USE_CALLING_UID);/*這裏的 Camera 爲本地的類,在下面步驟的Camera.cpp中,而不是前面用到的Camera.java中的類。*/
} else {
jint status =Camera::connectLegacy(cameraId, halVersion, clientName,
Camera::USE_CALLING_UID,camera);
if (status != NO_ERROR) {
return status;
}
}
if (camera == NULL) {
return -EACCES;
}
// make sure camera hardware is alive
if (camera->getStatus() != NO_ERROR) {
return NO_INIT;
}
jclass clazz =env->GetObjectClass(thiz);
if (clazz == NULL) {
// This should never happen
jniThrowRuntimeException(env,"Can't find android/hardware/Camera");
return INVALID_OPERATION;
}
// We use a weak reference so the Cameraobject can be garbage collected.
// The reference is only used as a proxyfor callbacks.
//!++
#if 0 // Use MTK JNI for MTK camerafeature.
sp<JNICameraContext> context = newJNICameraContext(env, weak_this, clazz, camera);
#else
sp<JNICameraContext> context = new MtkJNICameraContext(env,weak_this, clazz, camera);
#endif
//!--
context->incStrong((void*)android_hardware_Camera_native_setup);
camera->setListener(context);
// save context in opaque field
env->SetLongField(thiz, fields.context,(jlong)context.get());
return NO_ERROR;
}
STEP 6:
\frameworks\av\camera\Camera.cpp
sp<Camera>Camera::connect(int cameraId, const String16& clientPackageName,
int clientUid)
{
return CameraBaseT::connect(cameraId,clientPackageName, clientUid);
}
/*
CameraBaseT在/frameworks/av/include/camera/CameraBase.h中有定義以下:
typedefCameraBase<TCam> CameraBaseT
因此就是調用的CameraBase類的connect了。
*/
STEP 7:
\frameworks\av\camera\CameraBase.cpp
template<typename TCam, typename TCamTraits>
sp<TCam>CameraBase<TCam, TCamTraits>::connect(int cameraId,
const String16& clientPackageName,
int clientUid)
{
ALOGV("%s: connect",__FUNCTION__);
sp<TCam> c = new TCam(cameraId);
sp<TCamCallbacks> cl = c;
status_t status = NO_ERROR;
constsp<ICameraService>& cs = getCameraService();/*獲取cameraService,這個在開機的時候已經註冊過的一個camera服務: media.camera */
if (cs != 0) {
TCamConnectService fnConnectService =TCamTraits::fnConnectService;
status =(cs.get()->*fnConnectService)(cl, cameraId, clientPackageName, clientUid,
/*out*/ c->mCamera);
/*
這個對應的就是CameraService::connect;這裏面的具體轉換過程還不太清楚。待後續學習。Cs.get()是什麼意思?CameraService 中沒有找到get這個函數。
fnConnectService的定義在:\frameworks\av\camera\ Camera.cpp
CameraTraits<Camera>::TCamConnectServiceCameraTraits<Camera>::fnConnectService =
&ICameraService::connect;
*/
}
if (status == OK && c->mCamera!= 0) {
IInterface::asBinder(c->mCamera)->linkToDeath(c);
c->mStatus = NO_ERROR;
} else {
ALOGW("An error occurred whileconnecting to camera: %d", cameraId);
c.clear();
}
return c;
}
STEP 8:
\frameworks\av\services\camera\libcameraservice\CameraService.cpp
status_tCameraService::connect(
const sp<ICameraClient>&cameraClient,
int cameraId,
const String16& clientPackageName,
int clientUid,
/*out*/
sp<ICamera>& device) {
status_t ret = NO_ERROR;
String8 id =String8::format("%d", cameraId);
sp<Client> client = nullptr;
ret =connectHelper<ICameraClient,Client>(cameraClient, id,CAMERA_HAL_API_VERSION_UNSPECIFIED,
clientPackageName, clientUid, API_1, false, false, /*out*/client);
if(ret != NO_ERROR) {
logRejected(id, getCallingPid(),String8(clientPackageName),
String8::format("%s(%d)", strerror(-ret), ret));
return ret;
}
device = client;
return NO_ERROR;
}
STEP 9:
\frameworks\av\services\camera\libcameraservice\CameraService.h
template<classCALLBACK, class CLIENT>
status_t CameraService::connectHelper(constsp<CALLBACK>& cameraCb, const String8& cameraId,
int halVersion, const String16&clientPackageName, int clientUid,
apiLevel effectiveApiLevel, boollegacyMode, bool shimUpdateOnly,
/*out*/sp<CLIENT>& device) {
status_t ret = NO_ERROR;
String8 clientName8(clientPackageName);
int clientPid = getCallingPid();
ALOGI("CameraService::connect call(PID %d \"%s\", camera ID %s) for HAL version %s and "
"Camera API version %d",clientPid, clientName8.string(), cameraId.string(),
(halVersion == -1) ?"default" : std::to_string(halVersion).c_str(),
static_cast<int>(effectiveApiLevel));
sp<CLIENT> client = nullptr;
{
// Acquire mServiceLock and preventother clients from connecting
std::unique_ptr<AutoConditionLock>lock =
AutoConditionLock::waitAndAcquire(mServiceLockWrapper,DEFAULT_CONNECT_TIMEOUT_NS);
if (lock == nullptr) {
ALOGE("CameraService::connectX (PID %d) rejected (too many other clients connecting)."
, clientPid);
return -EBUSY;
}
// Enforce client permissions and dobasic sanity checks
if((ret =validateConnectLocked(cameraId, /*inout*/clientUid)) != NO_ERROR) {
return ret;
}
// Check the shim parameters afteracquiring lock, if they have already been updated and
// we were doing a shim update, returnimmediately
if (shimUpdateOnly) {
auto cameraState =getCameraState(cameraId);
if (cameraState != nullptr) {
if(!cameraState->getShimParams().isEmpty()) return NO_ERROR;
}
}
sp<BasicClient> clientTmp =nullptr;
std::shared_ptr<resource_policy::ClientDescriptor<String8,sp<BasicClient>>> partial;
if ((ret =handleEvictionsLocked(cameraId, clientPid, effectiveApiLevel,
IInterface::asBinder(cameraCb),clientName8, /*out*/&clientTmp,
/*out*/&partial)) !=NO_ERROR) {
return ret;
}
if (clientTmp.get() != nullptr) {
// Handle special case for API1MediaRecorder where the existing client is returned
device =static_cast<CLIENT*>(clientTmp.get());
return NO_ERROR;
}
// give flashlight a chance to closedevices if necessary.
mFlashlight->prepareDeviceOpen(cameraId);
// TODO: Update getDeviceVersion + HALinterface to use strings for Camera IDs
int id = cameraIdToInt(cameraId);
if (id == -1) {
ALOGE("%s: Invalid camera ID%s, cannot get device version from HAL.", __FUNCTION__,
cameraId.string());
return BAD_VALUE;
}
int facing = -1;
int deviceVersion =getDeviceVersion(id, /*out*/&facing);
sp<BasicClient> tmp = nullptr;
if((ret =makeClient(this, cameraCb, clientPackageName, cameraId, facing, clientPid,
clientUid, getpid(), legacyMode, halVersion, deviceVersion,effectiveApiLevel,
/*out*/&tmp)) != NO_ERROR) {
return ret;
}
/*
上面的makeClient建立的只是cameraservice內部類client,而不是客戶端。主要從打印中能夠看到,是先建立cameraservice的內部clinet而後再建立的客戶端clinet的。具體過程帶分析。
01-02 00:27:12.751 181 448 D CameraService:[CameraService::loadSound] + tid:448 mSoundLock - ref=0
01-02 00:27:12.751 181 448 D CameraService:CameraService::loadSound ref=0
01-02 00:27:12.751 638 660 V DisplayEventReceiver:call nativeScheduleVsync mReceiverPtr = -1606054400
01-02 00:27:12.752 181 448 D CameraService:Client::Client X (pid 4232, id 0)
01-02 00:27:12.752 181 448 D CameraClient:CameraClient::CameraClient E (pid 4232, id 0)
01-02 00:27:12.752 181 448 D CameraClient:CameraClient::CameraClient X (pid 4232, id 0)
01-02 00:27:12.752 181 448 D CameraClient:CameraClient::initialize E (pid 4232, id 0)
*/
client =static_cast<CLIENT*>(tmp.get());
/*
Client則爲客戶端,具體的實現帶分析。Tmp.get()不知道什麼意思?
*/
LOG_ALWAYS_FATAL_IF(client.get() ==nullptr, "%s: CameraService in invalid state",
__FUNCTION__);
if ((ret =client->initialize(mModule)) != OK) {/*客戶端初始化*/
/*
這裏的mModule就是開機初始化cameraservice實例的時候獲取的
mModule = newCameraModule(rawModule);
ALOGI("Loaded \"%s\" camera module",mModule->getModuleName());
err =mModule->init();
if (err != OK){
ALOGE("Could not initialize camera HAL module: %d (%s)", err,
strerror(-err));
logServiceError("Could not initialize camera HAL module",err);
mNumberOfCameras = 0;
deletemModule;
mModule =nullptr;
return;
}
*/
ALOGE("%s: Could notinitialize client from HAL module.", __FUNCTION__);
return ret;
}
//!++ The part is merged by MTK, googlewill merge in MR1
/*
sp<IBinder> remoteCallback =client->getRemote();
if (remoteCallback != nullptr) {
remoteCallback->linkToDeath(this);
}
*/
//!--
// Update shim paremeters for legacyclients
if (effectiveApiLevel == API_1) {
// Assume we have always received aClient subclass for API1
sp<Client> shimClient =reinterpret_cast<Client*>(client.get());
String8 rawParams =shimClient->getParameters();
CameraParameters params(rawParams);
auto cameraState =getCameraState(cameraId);
if (cameraState != nullptr) {
cameraState->setShimParams(params);
} else {
ALOGE("%s: Cannot updateshim parameters for camera %s, no such device exists.",
__FUNCTION__,cameraId.string());
}
}
if (shimUpdateOnly) {
// If only updating legacy shimparameters, immediately disconnect client
mServiceLock.unlock();
client->disconnect();
mServiceLock.lock();
} else {
// Otherwise, add client to activeclients list
finishConnectLocked(client,partial);
}
} // lock is destroyed, allow furtherconnect calls
// Important: release the mutex here so theclient can call back into the service from its
// destructor (can be at the end of thecall)
device = client;
return NO_ERROR;
}
STEP 10:
\frameworks\av\services\camera\libcameraservice\api1\CameraClient.cpp
status_tCameraClient::initialize(CameraModule *module) {
int callingPid = getCallingPid();
status_t res;
LOG1("CameraClient::initialize E (pid%d, id %d)", callingPid, mCameraId);
// Verify ops permissions
res = startCameraOps();
if (res != OK) {
return res;
}
char camera_device_name[10];
snprintf(camera_device_name,sizeof(camera_device_name), "%d", mCameraId);
mHardware = newCameraHardwareInterface(camera_device_name);
/*
mHardware 定義是 CameraHardwareInterface,他也是Android的通用接口。各個廠家提供的功能都要經過CameraHardwareInterface適配向CameraService提供硬件操做接口。
*/
res = mHardware->initialize(module);
if (res != OK) {
ALOGE("%s: Camera %d: unable toinitialize device: %s (%d)",
__FUNCTION__, mCameraId,strerror(-res), res);
mHardware.clear();
return res;
}
//!++
// mtk callback
mHardware->setMtkCallbacks(
mtkMetadataCallback,
(void *)(uintptr_t)mCameraId);
//!--
mHardware->setCallbacks(notifyCallback,
dataCallback,
dataCallbackTimestamp,
(void *)(uintptr_t)mCameraId);
// Enable zoom, error, focus, and metadatamessages by default
enableMsgType(CAMERA_MSG_ERROR |CAMERA_MSG_ZOOM | CAMERA_MSG_FOCUS |
CAMERA_MSG_PREVIEW_METADATA |CAMERA_MSG_FOCUS_MOVE);
//!++
enableMsgType(MTK_CAMERA_MSG_EXT_NOTIFY |MTK_CAMERA_MSG_EXT_DATA); // Enable MTK-extended messages by default
//!--
LOG1("CameraClient::initialize X (pid %d,id %d)", callingPid, mCameraId);
return OK;
}
STEP 10:
\frameworks\av\services\camera\libcameraservice\device1\CameraHardwareInterface.h
status_t initialize(CameraModule *module)
{
ALOGI("Opening camera %s",mName.string());
camera_info info;
status_t res = module->getCameraInfo(atoi(mName.string()),&info);
if (res != OK) return res;
int rc = OK;
if (module->getModuleApiVersion()>= CAMERA_MODULE_API_VERSION_2_3 &&
info.device_version >CAMERA_DEVICE_API_VERSION_1_0) {
// Open higher version cameradevice as HAL1.0 device.
rc =module->openLegacy(mName.string(),
CAMERA_DEVICE_API_VERSION_1_0,
(hw_device_t **)&mDevice);
} else {
rc =module->open(mName.string(), (hw_device_t **)&mDevice);
}
if (rc != OK) {
ALOGE("Could not open camera%s: %d", mName.string(), rc);
return rc;
}
initHalPreviewWindow();
return rc;
}
/*
module->open這裏的module就是\frameworks\av\services\camera\libcameraservice\CameraService.cpp中的onFirstRef在啓動該服務的時候就建立好了,而cameraserver是在開機的時候就servicemanager中啓動的。
void CameraService::onFirstRef()
{
ALOGI("CameraService process starting");
BnCameraService::onFirstRef();
// Updatebattery life tracking if service is restarting
BatteryNotifier& notifier(BatteryNotifier::getInstance());
notifier.noteResetCamera();
notifier.noteResetFlashlight();
camera_module_t*rawModule;
int err= hw_get_module(CAMERA_HARDWARE_MODULE_ID,
(consthw_module_t **)&rawModule);
/*
這裏的hw_get_module獲取的就是CAMERA_HARDWARE_MODULE_ID對應的庫文檔,對應咱們如今分析平臺就是camera.mt8127.so,mt8127對應的是MTK的芯片平臺。
*/
if (err < 0){
ALOGE("Could not load camera HAL module: %d (%s)", err,strerror(-err));
logServiceError("Could not load camera HAL module", err);
mNumberOfCameras = 0;
return;
}
mModule = newCameraModule(rawModule);
ALOGI("Loaded \"%s\" camera module",mModule->getModuleName());
err =mModule->init();
if (err != OK){
ALOGE("Could not initialize camera HAL module: %d (%s)", err,
strerror(-err));
logServiceError("Could not initialize camera HAL module",err);
mNumberOfCameras = 0;
deletemModule;
mModule =nullptr;
return;
}
mNumberOfCameras = mModule->getNumberOfCameras();
mNumberOfNormalCameras = mNumberOfCameras;
……
*/
HAL層就是camera.mt8127.so具體的實現,就是MTK作的一些接口。對於\frameworks\av\services\camera\libcameraservice\device1\CameraHardwareInterface.h具體的實現。因此後面一篇文章中,咱們再單獨的學習CAMERA 的 HAL.