日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

Android:高通平台Camera HFR Usecase分析

發布時間:2023/12/13 综合教程 32 生活家
生活随笔 收集整理的這篇文章主要介紹了 Android:高通平台Camera HFR Usecase分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、高幀率錄像簡介

  高幀率錄像即慢動作拍攝,通常人眼能夠接受的最好的視頻幀速率是24幀/每秒。如果用120幀/秒拍攝一個動作,再用24幀每秒來播放的話,視頻就放慢了5倍。

  高通平臺的 Slow motion feature :

高速錄制(HSR) : 以高fps(運行速率)捕獲、編碼并保存為高 fps(目標速率),運行速率等于目標速率。
高幀率錄制(HFR) : 以高fps(運行速率)捕獲、編碼并保存為30 fps(目標速率),運行速率大于目標速率。

、代碼流程分析 (高通相機源碼路徑:packagesappsSnapdragonCamera)

1、app啟動錄像 : packagesappsSnapdragonCamerasrccomandroidcameraCaptureModule.java

    private boolean startRecordingVideo(final int cameraId) {
            ...

            if (ApiHelper.isAndroidPOrHigher()) {
                if (mHighSpeedCapture && ((int) mHighSpeedFPSRange.getUpper() > NORMAL_SESSION_MAX_FPS)) {
                    CaptureRequest initialRequest = mVideoRequestBuilder.build();
                    buildConstrainedCameraSession(mCameraDevice[cameraId], surfaces,
                            mSessionListener, mCameraHandler, initialRequest);

                } else {
                    configureCameraSessionWithParameters(cameraId, surfaces,
                            mSessionListener, mCameraHandler, mVideoRequestBuilder.build());
                }
            } else {

                //hfr開啟且最大幀率大于NORMAL_SESSION_MAX_FPS(60fps)
                //創建的是createConstrainedHighSpeedCaptureSession
                //否則是createCaptureSession
                if (mHighSpeedCapture && ((int) mHighSpeedFPSRange.getUpper() > NORMAL_SESSION_MAX_FPS)) {
            //創建高速流 mCameraDevice[cameraId].createConstrainedHighSpeedCaptureSession(surfaces, new CameraConstrainedHighSpeedCaptureSession.StateCallback() { @Override public void onConfigured(CameraCaptureSession cameraCaptureSession) { mCurrentSession = cameraCaptureSession; Log.v(TAG, "createConstrainedHighSpeedCaptureSession onConfigured"); mCaptureSession[cameraId] = cameraCaptureSession; CameraConstrainedHighSpeedCaptureSession session = (CameraConstrainedHighSpeedCaptureSession) mCurrentSession;

try {
setUpVideoCaptureRequestBuilder(mVideoRequestBuilder, cameraId);
List list = CameraUtil
.createHighSpeedRequestList(mVideoRequestBuilder.build());
                        // 通過setRepeatingBurst每次同時提交多個request申請,對應的native方法是submitRequestList
session.setRepeatingBurst(list, mCaptureCallback, mCameraHandler);
} catch (CameraAccessException e) {
Log.e(TAG, "Failed to start high speed video recording "
+ e.getMessage());
e.printStackTrace();
} catch (IllegalArgumentException e) {
Log.e(TAG, "Failed to start high speed video recording "
+ e.getMessage());
e.printStackTrace();
} catch (IllegalStateException e) {
Log.e(TAG, "Failed to start high speed video recording "
+ e.getMessage());
e.printStackTrace();
}
if (!mFrameProcessor.isFrameListnerEnabled() && !startMediaRecorder()) {
startRecordingFailed();
return;
} }, null); } else { surfaces.add(mVideoSnapshotImageReader.getSurface()); String zzHDR = mSettingsManager.getValue(SettingsManager.KEY_VIDEO_HDR_VALUE); boolean zzHdrStatue = zzHDR.equals("1"); // if enable ZZHDR mode, don`t call the setOpModeForVideoStream method. if (!zzHdrStatue) { setOpModeForVideoStream(cameraId); } String value = mSettingsManager.getValue(SettingsManager.KEY_FOVC_VALUE); if (value != null && Boolean.parseBoolean(value)) { mStreamConfigOptMode = mStreamConfigOptMode | STREAM_CONFIG_MODE_FOVC; } if (zzHdrStatue) { mStreamConfigOptMode = STREAM_CONFIG_MODE_ZZHDR; } if (DEBUG) { Log.v(TAG, "createCustomCaptureSession mStreamConfigOptMode :" + mStreamConfigOptMode); } if (mStreamConfigOptMode == 0) {
               //普通流,但是該過程設置了setOpModeForVideoStream,會導致config->operation_mode變化。 mCameraDevice[cameraId].createCaptureSession(surfaces, mCCSSateCallback, null); } else { List<OutputConfiguration> outConfigurations = new ArrayList<>(surfaces.size()); for (Surface sface : surfaces) { outConfigurations.add(new OutputConfiguration(sface)); } mCameraDevice[cameraId].createCustomCaptureSession(null, outConfigurations, mStreamConfigOptMode, mCCSSateCallback, null); } } } } ...

2、HFR 配置流 : frameworksasecorejavaandroidhardwarecamera2CameraDevice.java

 //創建高速捕獲會話接口
public abstract void createConstrainedHighSpeedCaptureSession(@NonNull List<Surface> outputs, @NonNull CameraCaptureSession.StateCallback callback, @Nullable Handler handler) throws CameraAccessException;

  具體實現在:frameworksasecorejavaandroidhardwarecamera2implCameraDeviceImpl.java

    @Override
    public void createConstrainedHighSpeedCaptureSession(List<Surface> outputs,
            android.hardware.camera2.CameraCaptureSession.StateCallback callback, Handler handler)
            throws CameraAccessException {
        if (outputs == null || outputs.size() == 0 || outputs.size() > 2) {
            throw new IllegalArgumentException(
                    "Output surface list must not be null and the size must be no more than 2");
        }
        List<OutputConfiguration> outConfigurations = new ArrayList<>(outputs.size());
        for (Surface surface : outputs) {
            outConfigurations.add(new OutputConfiguration(surface));
        }
        createCaptureSessionInternal(null, outConfigurations, callback,
                checkAndWrapHandler(handler),
                /*operatingMode*/ICameraDeviceUser.CONSTRAINED_HIGH_SPEED_MODE,
                /*sessionParams*/ null);
    }

  其中 createCaptureSessionInternal 函數實現如下:

    private void createCaptureSessionInternal(InputConfiguration inputConfig,
            List<OutputConfiguration> outputConfigurations,
            CameraCaptureSession.StateCallback callback, Executor executor,
            int operatingMode, CaptureRequest sessionParams) throws CameraAccessException {
        synchronized(mInterfaceLock) {
            if (DEBUG) {
                Log.d(TAG, "createCaptureSessionInternal");
            }

            checkIfCameraClosedOrInError();

            boolean isConstrainedHighSpeed =
                    (operatingMode == ICameraDeviceUser.CONSTRAINED_HIGH_SPEED_MODE);
            if (isConstrainedHighSpeed && inputConfig != null) {
                throw new IllegalArgumentException("Constrained high speed session doesn't support"
                        + " input configuration yet.");
            }

            // Notify current session that it's going away, before starting camera operations
            // After this call completes, the session is not allowed to call into CameraDeviceImpl
            if (mCurrentSession != null) {
                mCurrentSession.replaceSessionClose();
            }

            // TODO: dont block for this
            boolean configureSuccess = true;
            CameraAccessException pendingException = null;
            Surface input = null;
            try {
                // configure streams and then block until IDLE
                // 里面會獲取設備屬性
                configureSuccess = configureStreamsChecked(inputConfig, outputConfigurations,
                        operatingMode, sessionParams);
                if (configureSuccess == true && inputConfig != null) {
                    input = mRemoteDevice.getInputSurface();
                }
            } catch (CameraAccessException e) {
                configureSuccess = false;
                pendingException = e;
                input = null;
                if (DEBUG) {
                    Log.v(TAG, "createCaptureSession - failed with exception ", e);
                }
            }

            // Fire onConfigured if configureOutputs succeeded, fire onConfigureFailed otherwise.
            CameraCaptureSessionCore newSession = null;
            if (isConstrainedHighSpeed) {
                ArrayList<Surface> surfaces = new ArrayList<>(outputConfigurations.size());
                for (OutputConfiguration outConfig : outputConfigurations) {
                    surfaces.add(outConfig.getSurface());
                }
                StreamConfigurationMap config =
                    getCharacteristics().get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP);

                // 檢查格式是否正確、fps是否是有效范圍、是不是預覽/錄像編碼流等
                SurfaceUtils.checkConstrainedHighSpeedSurfaces(surfaces, /*fpsRange*/null, config);

                newSession = new CameraConstrainedHighSpeedCaptureSessionImpl(mNextSessionId++,
                        callback, executor, this, mDeviceExecutor, configureSuccess,
                        mCharacteristics);
            } else {
                newSession = new CameraCaptureSessionImpl(mNextSessionId++, input,
                        callback, executor, this, mDeviceExecutor, configureSuccess);
            }

            // TODO: wait until current session closes, then create the new session
            mCurrentSession = newSession;

            if (pendingException != null) {
                throw pendingException;
            }

            mSessionStateCallback = mCurrentSession.getDeviceStateCallback();
        }
    }

  繼續分析configureStreamsChecked()

    public boolean configureStreamsChecked(InputConfiguration inputConfig,
            List<OutputConfiguration> outputs, int operatingMode, CaptureRequest sessionParams)
                    throws CameraAccessException {
        // Treat a null input the same an empty list
        if (outputs == null) {
            outputs = new ArrayList<OutputConfiguration>();
        }
        if (outputs.size() == 0 && inputConfig != null) {
            throw new IllegalArgumentException("cannot configure an input stream without " +
                    "any output streams");
        }

        checkInputConfiguration(inputConfig);

        boolean success = false;

        synchronized(mInterfaceLock) {
            checkIfCameraClosedOrInError();
            // Streams to create
            HashSet<OutputConfiguration> addSet = new HashSet<OutputConfiguration>(outputs);
            // Streams to delete
            List<Integer> deleteList = new ArrayList<Integer>();

            // Determine which streams need to be created, which to be deleted
            for (int i = 0; i < mConfiguredOutputs.size(); ++i) {
                int streamId = mConfiguredOutputs.keyAt(i);
                OutputConfiguration outConfig = mConfiguredOutputs.valueAt(i);

                if (!outputs.contains(outConfig) || outConfig.isDeferredConfiguration()) {
                    // Always delete the deferred output configuration when the session
                    // is created, as the deferred output configuration doesn't have unique surface
                    // related identifies.
                    deleteList.add(streamId);
                } else {
                    addSet.remove(outConfig);  // Don't create a stream previously created
                }
            }

            mDeviceExecutor.execute(mCallOnBusy);
            stopRepeating();

            try {
                waitUntilIdle();

                // 開始配置
                mRemoteDevice.beginConfigure();

                // reconfigure the input stream if the input configuration is different.
                InputConfiguration currentInputConfig = mConfiguredInput.getValue();
                if (inputConfig != currentInputConfig &&
                        (inputConfig == null || !inputConfig.equals(currentInputConfig))) {
                    if (currentInputConfig != null) {
                        mRemoteDevice.deleteStream(mConfiguredInput.getKey());
                        mConfiguredInput = new SimpleEntry<Integer, InputConfiguration>(
                                REQUEST_ID_NONE, null);
                    }
                    if (inputConfig != null) {
                        int streamId = mRemoteDevice.createInputStream(inputConfig.getWidth(),
                                inputConfig.getHeight(), inputConfig.getFormat());
                        mConfiguredInput = new SimpleEntry<Integer, InputConfiguration>(
                                streamId, inputConfig);
                    }
                }

                // Delete all streams first (to free up HW resources)
                for (Integer streamId : deleteList) {
                    mRemoteDevice.deleteStream(streamId);
                    mConfiguredOutputs.delete(streamId);
                }

                // Add all new streams
                for (OutputConfiguration outConfig : outputs) {
                    if (addSet.contains(outConfig)) {
                        int streamId = mRemoteDevice.createStream(outConfig);
                        mConfiguredOutputs.put(streamId, outConfig);
                    }
                }

                //customOpMode 可以通過setOpModeForVideoStream改變
                //CameraConstrainedHighSpeedCaptureSessionImpl沒有改變該值
                operatingMode = (operatingMode | (customOpMode << 16));

                //結束配置流
                //mRemoteDevice類型是ICameraDeviceUserWrapper
                //是在在打開相機時獲取的。
                if (sessionParams != null) {
                    mRemoteDevice.endConfigure(operatingMode, sessionParams.getNativeCopy());
                } else {
                    mRemoteDevice.endConfigure(operatingMode, null);
                }

                success = true;
            } catch (IllegalArgumentException e) {
                // OK. camera service can reject stream config if it's not supported by HAL
                // This is only the result of a programmer misusing the camera2 api.
                Log.w(TAG, "Stream configuration failed due to: " + e.getMessage());
                return false;
            } catch (CameraAccessException e) {
                if (e.getReason() == CameraAccessException.CAMERA_IN_USE) {
                    throw new IllegalStateException("The camera is currently busy." +
                            " You must wait until the previous operation completes.", e);
                }
                throw e;
            } finally {
                if (success && outputs.size() > 0) {
                    mDeviceExecutor.execute(mCallOnIdle);
                } else {
                    // Always return to the 'unconfigured' state if we didn't hit a fatal error
                    mDeviceExecutor.execute(mCallOnUnconfigured);
                }
            }
        }

        return success;
    }

  上面的mRemoteDevice對象是ICameraDeviceUserWrapper類型,是在打開相機時獲取的,代碼如下:

  frameworksasecorejavaandroidhardwarecamera2CameraManager.java

    private CameraDevice openCameraDeviceUserAsync(String cameraId,
            CameraDevice.StateCallback callback, Executor executor, final int uid)
            throws CameraAccessException {
        CameraCharacteristics characteristics = getCameraCharacteristics(cameraId);
        CameraDevice device = null;

        synchronized (mLock) {

            ICameraDeviceUser cameraUser = null;

            //創建CameraDeviceImpl對象
            android.hardware.camera2.impl.CameraDeviceImpl deviceImpl =
                    new android.hardware.camera2.impl.CameraDeviceImpl(
                        cameraId,
                        callback,
                        executor,
                        characteristics,
                        mContext.getApplicationInfo().targetSdkVersion);

            ICameraDeviceCallbacks callbacks = deviceImpl.getCallbacks();

            try {
                if (supportsCamera2ApiLocked(cameraId)) {
                    // Use cameraservice's cameradeviceclient implementation for HAL3.2+ devices
                    //獲取cameraService代理對象
                    ICameraService cameraService = CameraManagerGlobal.get().getCameraService();
                    if (cameraService == null) {
                        throw new ServiceSpecificException(
                            ICameraService.ERROR_DISCONNECTED,
                            "Camera service is currently unavailable");
                    }
                    //通過cameraService代理對象打開相機,獲取ICameraDeviceUser cameraUser對象
                    cameraUser = cameraService.connectDevice(callbacks, cameraId,
                            mContext.getOpPackageName(), uid);
                } else {
                    // Use legacy camera implementation for HAL1 devices
                    int id;
                    try {
                        id = Integer.parseInt(cameraId);
                    } catch (NumberFormatException e) {
                        throw new IllegalArgumentException("Expected cameraId to be numeric, but it was: "
                                + cameraId);
                    }

                    Log.i(TAG, "Using legacy camera HAL.");
                    cameraUser = CameraDeviceUserShim.connectBinderShim(callbacks, id);
                }
            } catch (ServiceSpecificException e) {
                if (e.errorCode == ICameraService.ERROR_DEPRECATED_HAL) {
                    throw new AssertionError("Should've gone down the shim path");
                } else if (e.errorCode == ICameraService.ERROR_CAMERA_IN_USE ||
                        e.errorCode == ICameraService.ERROR_MAX_CAMERAS_IN_USE ||
                        e.errorCode == ICameraService.ERROR_DISABLED ||
                        e.errorCode == ICameraService.ERROR_DISCONNECTED ||
                        e.errorCode == ICameraService.ERROR_INVALID_OPERATION) {
                    // Received one of the known connection errors
                    // The remote camera device cannot be connected to, so
                    // set the local camera to the startup error state
                    deviceImpl.setRemoteFailure(e);

                    if (e.errorCode == ICameraService.ERROR_DISABLED ||
                            e.errorCode == ICameraService.ERROR_DISCONNECTED ||
                            e.errorCode == ICameraService.ERROR_CAMERA_IN_USE) {
                        // Per API docs, these failures call onError and throw
                        throwAsPublicException(e);
                    }
                } else {
                    // Unexpected failure - rethrow
                    throwAsPublicException(e);
                }
            } catch (RemoteException e) {
                // Camera service died - act as if it's a CAMERA_DISCONNECTED case
                ServiceSpecificException sse = new ServiceSpecificException(
                    ICameraService.ERROR_DISCONNECTED,
                    "Camera service is currently unavailable");
                deviceImpl.setRemoteFailure(sse);
                throwAsPublicException(sse);
            }

            // TODO: factor out callback to be non-nested, then move setter to constructor
            // For now, calling setRemoteDevice will fire initial
            // onOpened/onUnconfigured callbacks.
            // This function call may post onDisconnected and throw CAMERA_DISCONNECTED if
            // cameraUser dies during setup.
            //將打開相機獲取的cameraUser對象設置到CameraDeviceImpl deviceImpl對象中
            deviceImpl.setRemoteDevice(cameraUser);
            device = deviceImpl;
        }
        
        //返回CameraDeviceImpl對象deviceImpl
        return device;
    }

 前面分析configureStreamsChecked可知其主要分為3步:

mRemoteDevice.beginConfigure();//開始配置
mRemoteDevice.deleteStream(streamId)和mRemoteDevice.createStream(outConfig)//創建、刪除流
mRemoteDevice.endConfigure(operatingMode);//結束配置,前兩步是準備工作,這一步才是真正配置流

主要分析下endConfigure:
  frameworksasecorejavaandroidhardwarecamera2implICameraDeviceUserWrapper.java

    public void endConfigure(int operatingMode, CameraMetadataNative sessionParams)
           throws CameraAccessException {
        try {
            // 通過Binder IPC 實際調用接口實現在 CameraDeviceClient.cpp
            mRemoteDevice.endConfigure(operatingMode, (sessionParams == null) ?
                    new CameraMetadataNative() : sessionParams);
        } catch (Throwable t) {
            CameraManager.throwAsPublicException(t);
            throw new UnsupportedOperationException("Unexpected exception", t);
        }
    }

  進入 frameworksavservicescameralibcameraserviceapi2CameraDeviceClient.cpp

binder::Status CameraDeviceClient::endConfigure(int operatingMode,
        const hardware::camera2::impl::CameraMetadataNative& sessionParams) {
    ATRACE_CALL();
    ALOGV("%s: ending configure (%d input stream, %zu output surfaces)",
            __FUNCTION__, mInputStream.configured ? 1 : 0,
            mStreamMap.size());
......
   // Sanitize the high speed session against necessary capability bit. bool isConstrainedHighSpeed = (operatingMode == ICameraDeviceUser::CONSTRAINED_HIGH_SPEED_MODE); // 檢查是否支持CONSTRAINED_HIGH_SPEED_MODE if (isConstrainedHighSpeed) { CameraMetadata staticInfo = mDevice->info(); camera_metadata_entry_t entry = staticInfo.find(ANDROID_REQUEST_AVAILABLE_CAPABILITIES); bool isConstrainedHighSpeedSupported = false; for(size_t i = 0; i < entry.count; ++i) { uint8_t capability = entry.data.u8[i]; if (capability == ANDROID_REQUEST_AVAILABLE_CAPABILITIES_CONSTRAINED_HIGH_SPEED_VIDEO) { isConstrainedHighSpeedSupported = true; break; } } if (!isConstrainedHighSpeedSupported) { String8 msg = String8::format( "Camera %s: Try to create a constrained high speed configuration on a device" " that doesn't support it.", mCameraIdStr.string()); ALOGE("%s: %s", __FUNCTION__, msg.string()); return STATUS_ERROR(CameraService::ERROR_ILLEGAL_ARGUMENT, msg.string()); } } //檢查通過后開始配置流 status_t err = mDevice->configureStreams(sessionParams, operatingMode); if (err == BAD_VALUE) { String8 msg = String8::format("Camera %s: Unsupported set of inputs/outputs provided", mCameraIdStr.string()); ALOGE("%s: %s", __FUNCTION__, msg.string()); res = STATUS_ERROR(CameraService::ERROR_ILLEGAL_ARGUMENT, msg.string()); } else if (err != OK) { String8 msg = String8::format("Camera %s: Error configuring streams: %s (%d)", mCameraIdStr.string(), strerror(-err), err); ALOGE("%s: %s", __FUNCTION__, msg.string()); res = STATUS_ERROR(CameraService::ERROR_INVALID_OPERATION, msg.string()); } return res; }

 接著進入到:frameworksavservicescameralibcameraservicedevice3Camera3Device.cpp

status_t Camera3Device::configureStreams(const CameraMetadata& sessionParams, int operatingMode) {
    ATRACE_CALL();
    ALOGV("%s: E", __FUNCTION__);

    Mutex::Autolock il(mInterfaceLock);
    Mutex::Autolock l(mLock);

    // In case the client doesn't include any session parameter, try a
    // speculative configuration using the values from the last cached
    // default request.
    if (sessionParams.isEmpty() &&
            ((mLastTemplateId > 0) && (mLastTemplateId < CAMERA3_TEMPLATE_COUNT)) &&
            (!mRequestTemplateCache[mLastTemplateId].isEmpty())) {
        ALOGV("%s: Speculative session param configuration with template id: %d", __func__,
                mLastTemplateId);
        return filterParamsAndConfigureLocked(mRequestTemplateCache[mLastTemplateId],
                operatingMode);
    }

    return filterParamsAndConfigureLocked(sessionParams, operatingMode);
}

 接著調用到如下函數:

status_t Camera3Device::configureStreamsLocked(int operatingMode,
        const CameraMetadata& sessionParams, bool notifyRequestThread) {
    ATRACE_CALL();
    status_t res;

    if (mStatus != STATUS_UNCONFIGURED && mStatus != STATUS_CONFIGURED) {
        CLOGE("Not idle");
        return INVALID_OPERATION;
    }

    if (operatingMode < 0) {
        CLOGE("Invalid operating mode: %d", operatingMode);
        return BAD_VALUE;
    }

    // 檢查是否是isConstrainedHighSpeed模式
    bool isConstrainedHighSpeed =
            static_cast<int>(StreamConfigurationMode::CONSTRAINED_HIGH_SPEED_MODE) ==
            operatingMode;

    if (mOperatingMode != operatingMode) {
        mNeedConfig = true;
        mIsConstrainedHighSpeedConfiguration = isConstrainedHighSpeed;
        mOperatingMode = operatingMode;
    }

    if (!mNeedConfig) {
        ALOGV("%s: Skipping config, no stream changes", __FUNCTION__);
        return OK;
    }

    // Workaround for device HALv3.2 or older spec bug - zero streams requires
    // adding a dummy stream instead.
    // TODO: Bug: 17321404 for fixing the HAL spec and removing this workaround.
    if (mOutputStreams.size() == 0) {
        addDummyStreamLocked();
    } else {
        tryRemoveDummyStreamLocked();
    }

    // Start configuring the streams
    ALOGV("%s: Camera %s: Starting stream configuration", __FUNCTION__, mId.string());

    mPreparerThread->pause();

    camera3_stream_configuration config;
    config.operation_mode = mOperatingMode; //將mOperatingMode賦值給config.operation_mode
    config.num_streams = (mInputStream != NULL) + mOutputStreams.size();

    Vector<camera3_stream_t*> streams;
    streams.setCapacity(config.num_streams);
    std::vector<uint32_t> bufferSizes(config.num_streams, 0);


    if (mInputStream != NULL) {
        camera3_stream_t *inputStream;
        inputStream = mInputStream->startConfiguration();
        if (inputStream == NULL) {
            CLOGE("Can't start input stream configuration");
            cancelStreamsConfigurationLocked();
            return INVALID_OPERATION;
        }
        streams.add(inputStream);
    }

    // 輸出流配置
    for (size_t i = 0; i < mOutputStreams.size(); i++) {

        // Don't configure bidi streams twice, nor add them twice to the list
        if (mOutputStreams[i].get() ==
            static_cast<Camera3StreamInterface*>(mInputStream.get())) {

            config.num_streams--;
            continue;
        }

        camera3_stream_t *outputStream;
        outputStream = mOutputStreams.editValueAt(i)->startConfiguration();
        if (outputStream == NULL) {
            CLOGE("Can't start output stream configuration");
            cancelStreamsConfigurationLocked();
            return INVALID_OPERATION;
        }
        streams.add(outputStream);

        if (outputStream->format == HAL_PIXEL_FORMAT_BLOB &&
                outputStream->data_space == HAL_DATASPACE_V0_JFIF) {
            size_t k = i + ((mInputStream != nullptr) ? 1 : 0); // Input stream if present should
                                                                // always occupy the initial entry.
            bufferSizes[k] = static_cast<uint32_t>(
                    getJpegBufferSize(outputStream->width, outputStream->height));
        }
    }

    config.streams = streams.editArray();

    // Do the HAL configuration; will potentially touch stream
    // max_buffers, usage, priv fields.

    const camera_metadata_t *sessionBuffer = sessionParams.getAndLock();

    //通知HAL層配置流
    res = mInterface->configureStreams(sessionBuffer, &config, bufferSizes);
    sessionParams.unlock(sessionBuffer);
......
  return OK; }

 最后調用HIDL接口: frameworksavservicescameralibcameraservicedevice3Camera3Device.cpp

status_t Camera3Device::HalInterface::configureStreams(const camera_metadata_t *sessionParams,
        camera3_stream_configuration *config, const std::vector<uint32_t>& bufferSizes) {
    ATRACE_NAME("CameraHal::configureStreams");
    if (!valid()) return INVALID_OPERATION;
    status_t res = OK;

......
  // See if we have v3.4 or v3.3 HAL if (mHidlSession_3_4 != nullptr) { // We do; use v3.4 for the call ALOGV("%s: v3.4 device found", __FUNCTION__); device::V3_4::HalStreamConfiguration finalConfiguration3_4; auto err = mHidlSession_3_4->configureStreams_3_4(requestedConfiguration3_4, [&status, &finalConfiguration3_4] (common::V1_0::Status s, const device::V3_4::HalStreamConfiguration& halConfiguration) { finalConfiguration3_4 = halConfiguration; status = s; }); if (!err.isOk()) { ALOGE("%s: Transaction error: %s", __FUNCTION__, err.description().c_str()); return DEAD_OBJECT; } finalConfiguration.streams.resize(finalConfiguration3_4.streams.size()); for (size_t i = 0; i < finalConfiguration3_4.streams.size(); i++) { finalConfiguration.streams[i] = finalConfiguration3_4.streams[i].v3_3; } } else if (mHidlSession_3_3 != nullptr) { // We do; use v3.3 for the call ALOGV("%s: v3.3 device found", __FUNCTION__); auto err = mHidlSession_3_3->configureStreams_3_3(requestedConfiguration3_2, [&status, &finalConfiguration] (common::V1_0::Status s, const device::V3_3::HalStreamConfiguration& halConfiguration) { finalConfiguration = halConfiguration; status = s; }); if (!err.isOk()) { ALOGE("%s: Transaction error: %s", __FUNCTION__, err.description().c_str()); return DEAD_OBJECT; } } else { // We don't; use v3.2 call and construct a v3.3 HalStreamConfiguration ALOGV("%s: v3.2 device found", __FUNCTION__); HalStreamConfiguration finalConfiguration_3_2; auto err = mHidlSession->configureStreams(requestedConfiguration3_2, [&status, &finalConfiguration_3_2] (common::V1_0::Status s, const HalStreamConfiguration& halConfiguration) { finalConfiguration_3_2 = halConfiguration; status = s; }); if (!err.isOk()) { ALOGE("%s: Transaction error: %s", __FUNCTION__, err.description().c_str()); return DEAD_OBJECT; } finalConfiguration.streams.resize(finalConfiguration_3_2.streams.size()); for (size_t i = 0; i < finalConfiguration_3_2.streams.size(); i++) { finalConfiguration.streams[i].v3_2 = finalConfiguration_3_2.streams[i]; finalConfiguration.streams[i].overrideDataSpace = requestedConfiguration3_2.streams[i].dataSpace; } } ......
  return res; }

 HIDL接口各版本在: hardwareinterfacescameradevice

 HAL層的代碼部分如下: vendorqcomproprietarycamxsrccorehalcamxhal3.cpp

static int configure_streams(
    const struct camera3_device*    pCamera3DeviceAPI,
    camera3_stream_configuration_t* pStreamConfigsAPI)
{
    CAMX_ENTRYEXIT_SCOPE(CamxLogGroupHAL, SCOPEEventHAL3ConfigureStreams);
......

        Camera3StreamConfig* pStreamConfigs = reinterpret_cast<Camera3StreamConfig*>(pStreamConfigsAPI);

        result = pHALDevice->ConfigureStreams(pStreamConfigs);

        if ((CamxResultSuccess != result) && (CamxResultEInvalidArg != result))
        {
            // HAL interface requires -ENODEV (EFailed) if a fatal error occurs
            result = CamxResultEFailed;
        }
        if (CamxResultSuccess == result)
        {
            for (UINT32 stream = 0; stream < pStreamConfigsAPI->num_streams; stream++)
            {
                CAMX_ASSERT(NULL != pStreamConfigsAPI->streams[stream]);

                if (NULL == pStreamConfigsAPI->streams[stream])
                {
                    CAMX_LOG_ERROR(CamxLogGroupHAL, "Invalid argument 2 for configure_streams()");
                    // HAL interface requires -EINVAL (EInvalidArg) for invalid arguments
                    result = CamxResultEInvalidArg;
                    break;
                }
                else
                {
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, " FINAL stream[%d] = %p - info:", stream,
                        pStreamConfigsAPI->streams[stream]);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            format       : %d, %s",
                        pStreamConfigsAPI->streams[stream]->format,
                        FormatToString(pStreamConfigsAPI->streams[stream]->format));
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            width        : %d",
                        pStreamConfigsAPI->streams[stream]->width);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            height       : %d",
                        pStreamConfigsAPI->streams[stream]->height);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            stream_type  : %08x, %s",
                        pStreamConfigsAPI->streams[stream]->stream_type,
                        StreamTypeToString(pStreamConfigsAPI->streams[stream]->stream_type));
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            usage        : %08x",
                        pStreamConfigsAPI->streams[stream]->usage);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            max_buffers  : %d",
                        pStreamConfigsAPI->streams[stream]->max_buffers);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            rotation     : %08x, %s",
                        pStreamConfigsAPI->streams[stream]->rotation,
                        RotationToString(pStreamConfigsAPI->streams[stream]->rotation));
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            data_space   : %08x, %s",
                        pStreamConfigsAPI->streams[stream]->data_space,
                        DataSpaceToString(pStreamConfigsAPI->streams[stream]->data_space));
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            priv         : %p",
                        pStreamConfigsAPI->streams[stream]->priv);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            reserved[0]         : %p",
                        pStreamConfigsAPI->streams[stream]->reserved[0]);
                    CAMX_LOG_CONFIG(CamxLogGroupHAL, "            reserved[1]         : %p",
                        pStreamConfigsAPI->streams[stream]->reserved[1]);

                    Camera3HalStream* pHalStream =
                        reinterpret_cast<Camera3HalStream*>(pStreamConfigsAPI->streams[stream]->reserved[0]);
                    if (pHalStream != NULL)
                    {
                        if (TRUE == HwEnvironment::GetInstance()->GetStaticSettings()->enableHALFormatOverride) //GetInstance()中會初始化當前設備、sensor的 capabilities
                        {
                            pStreamConfigsAPI->streams[stream]->format =
                                static_cast<HALPixelFormat>(pHalStream->overrideFormat);
                        }
                        CAMX_LOG_CONFIG(CamxLogGroupHAL,
                            "   pHalStream: %p format : 0x%x, overrideFormat : 0x%x consumer usage: %llx, producer usage: %llx",
                            pHalStream, pStreamConfigsAPI->streams[stream]->format,
                            pHalStream->overrideFormat, pHalStream->consumerUsage, pHalStream->producerUsage);
                    }
                }
            }
        }
 ......
  return Utils::CamxResultToErrno(result);
}

 其中ConfigureStreams的實現在 vendorqcomproprietarycamxsrccorehalcamxhaldevice.cpp :

CamxResult HALDevice::ConfigureStreams(
    Camera3StreamConfig* pStreamConfigs)
{
    CamxResult result = CamxResultSuccess;

    // Validate the incoming stream configurations
    result = CheckValidStreamConfig(pStreamConfigs);

......
  if (CamxResultSuccess == result) { ClearFrameworkRequestBuffer(); m_numPipelines = 0; if (TRUE == m_bCHIModuleInitialized) { GetCHIAppCallbacks()->chi_teardown_override_session(reinterpret_cast<camera3_device*>(&m_camera3Device), 0, NULL); }
m_bCHIModuleInitialized = CHIModuleInitialize(pStreamConfigs); // 初始化Chi模塊 ...... } return result; }

  CHIModuleInitialize中調用了Chi層注冊的回調函數:

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// HALDevice::CHIModuleInitialize
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
BOOL HALDevice::CHIModuleInitialize(
    Camera3StreamConfig* pStreamConfigs)
{
    BOOL isOverrideEnabled = FALSE;

    if (TRUE == HAL3Module::GetInstance()->IsCHIOverrideModulePresent())
    {
        /// @todo (CAMX-1518) Handle private data from Override module
        VOID*                   pPrivateData;
        chi_hal_callback_ops_t* pCHIAppCallbacks  = GetCHIAppCallbacks();
     
     // 調用Chi層的回調函數
pCHIAppCallbacks->chi_initialize_override_session(GetCameraId(), reinterpret_cast<const camera3_device_t*>(&m_camera3Device), &m_HALCallbacks, reinterpret_cast<camera3_stream_configuration_t*>(pStreamConfigs), &isOverrideEnabled, &pPrivateData); } return isOverrideEnabled; }

 chi層的回調函數實現在 vendorqcomproprietarychi-cdkvendorchioverridedefaultchxextensioninterface.cpp :

CDKResult ExtensionModule::InitializeOverrideSession(
    uint32_t                        logicalCameraId,
    const camera3_device_t*         pCamera3Device,
    const chi_hal_ops_t*            chiHalOps,
    camera3_stream_configuration_t* pStreamConfig,
    int*                            pIsOverrideEnabled,
    VOID**                          pPrivate)
{
    CDKResult          result             = CDKResultSuccess;
    UINT32             modeCount          = 0;
    ChiSensorModeInfo* pAllModes          = NULL;
    UINT32             fps                = *m_pDefaultMaxFPS;
    BOOL               isVideoMode        = FALSE;
    uint32_t           operation_mode;
    static BOOL        fovcModeCheck      = EnableFOVCUseCase();
    UsecaseId          selectedUsecaseId  = UsecaseId::NoMatch;
    UINT               minSessionFps      = 0;
    UINT               maxSessionFps      = 0;
......
if ((isVideoMode == TRUE) && (operation_mode != 0)) { UINT32 numSensorModes = m_logicalCameraInfo[logicalCameraId].m_cameraCaps.numSensorModes; // 獲取sensor的信息,跟HFR相關的有 frameRate、batchedFrames等等 CHISENSORMODEINFO* pAllSensorModes = m_logicalCameraInfo[logicalCameraId].pSensorModeInfo; if ((operation_mode - 1) >= numSensorModes) { result = CDKResultEOverflow; CHX_LOG_ERROR("operation_mode: %d, numSensorModes: %d", operation_mode, numSensorModes); } else { fps = pAllSensorModes[operation_mode - 1].frameRate; } } if (CDKResultSuccess == result) { #if defined(CAMX_ANDROID_API) && (CAMX_ANDROID_API >= 28) //Android-P or better camera_metadata_t *metadata = const_cast<camera_metadata_t*>(pStreamConfig->session_parameters); camera_metadata_entry_t entry = { 0 }; entry.tag = ANDROID_CONTROL_AE_TARGET_FPS_RANGE; // The client may choose to send NULL sesssion parameter, which is fine. For example, torch mode // will have NULL session param. if (metadata != NULL) { // 獲取對應tag的entry結構體,并將數據保存在entry傳入的參數中。 int ret = find_camera_metadata_entry(metadata, entry.tag, &entry); if(ret == 0) { minSessionFps = entry.data.i32[0]; maxSessionFps = entry.data.i32[1]; m_usecaseMaxFPS = maxSessionFps; } } #endif if ((StreamConfigModeConstrainedHighSpeed == pStreamConfig->operation_mode) || (StreamConfigModeSuperSlowMotionFRC == pStreamConfig->operation_mode)) { // 如果是HFR模式則進行如下操作: // 1)查找與Video/Preview stream匹配的HFRVideoSizes。 // 注:preview 和 recording streams size必須一樣,否則高速攝像 session會創建失敗。 // 2)如果single entry在SupportedHFRVideoSizes中找到,我們就選擇這個entry中的batchsize。 SearchNumBatchedFrames(logicalCameraId, pStreamConfig, &m_usecaseNumBatchedFrames, &m_usecaseMaxFPS, maxSessionFps); if (480 > m_usecaseMaxFPS) { m_CurrentpowerHint = PERF_LOCK_POWER_HINT_VIDEO_ENCODE_HFR; } else { // For 480FPS or higher, require more aggresive power hint m_CurrentpowerHint = PERF_LOCK_POWER_HINT_VIDEO_ENCODE_HFR_480FPS; } } else { // Not a HFR usecase, batch frames value need to be set to 1. m_usecaseNumBatchedFrames = 1; if (maxSessionFps == 0) { m_usecaseMaxFPS = fps; } if (TRUE == isVideoMode) { if (30 >= m_usecaseMaxFPS) { m_CurrentpowerHint = PERF_LOCK_POWER_HINT_VIDEO_ENCODE; } else { m_CurrentpowerHint = PERF_LOCK_POWER_HINT_VIDEO_ENCODE_60FPS; } } else { m_CurrentpowerHint = PERF_LOCK_POWER_HINT_PREVIEW; } } if ((NULL != m_pPerfLockManager[logicalCameraId]) && (m_CurrentpowerHint != m_previousPowerHint)) { m_pPerfLockManager[logicalCameraId]->ReleasePerfLock(m_previousPowerHint); } // Example [B == batch]: (240 FPS / 4 FPB = 60 BPS) / 30 FPS (Stats frequency goal) = 2 BPF i.e. skip every other stats *m_pStatsSkipPattern = m_usecaseMaxFPS / m_usecaseNumBatchedFrames / 30; if (*m_pStatsSkipPattern < 1) { *m_pStatsSkipPattern = 1; } m_VideoHDRMode = (StreamConfigModeVideoHdr == pStreamConfig->operation_mode); m_torchWidgetUsecase = (StreamConfigModeQTITorchWidget == pStreamConfig->operation_mode); // this check is introduced to avoid set *m_pEnableFOVC == 1 if fovcEnable is disabled in // overridesettings & fovc bit is set in operation mode. // as well as to avoid set,when we switch Usecases. if (TRUE == fovcModeCheck) { *m_pEnableFOVC = ((pStreamConfig->operation_mode & StreamConfigModeQTIFOVC) == StreamConfigModeQTIFOVC) ? 1 : 0; } SetHALOps(chiHalOps, logicalCameraId); m_logicalCameraInfo[logicalCameraId].m_pCamera3Device = pCamera3Device; // 根據CameraInfo找到匹配Usecase selectedUsecaseId = m_pUsecaseSelector->GetMatchingUsecase(&m_logicalCameraInfo[logicalCameraId], pStreamConfig); CHX_LOG_CONFIG("Session_parameters FPS range %d:%d, BatchSize: %u FPS: %u SkipPattern: %u, " "cameraId = %d selected use case = %d", minSessionFps, maxSessionFps, m_usecaseNumBatchedFrames, m_usecaseMaxFPS, *m_pStatsSkipPattern, logicalCameraId, selectedUsecaseId); // FastShutter mode supported only in ZSL usecase. if ((pStreamConfig->operation_mode == StreamConfigModeFastShutter) && (UsecaseId::PreviewZSL != selectedUsecaseId)) { pStreamConfig->operation_mode = StreamConfigModeNormal; } m_operationMode[logicalCameraId] = pStreamConfig->operation_mode; } if (UsecaseId::NoMatch != selectedUsecaseId) { // 根據UsecaseId創建Usecase對象,HFR的UsecaseId是default m_pSelectedUsecase[logicalCameraId] = m_pUsecaseFactory->CreateUsecaseObject(&m_logicalCameraInfo[logicalCameraId], selectedUsecaseId, pStreamConfig);
   } .....

return result; }

  其中會調用到 AdvancedCameraUsecase::Create,實現在 vendorqcomproprietarychi-cdkvendorchioverridedefaultchxadvancedcamerausecase.cpp

AdvancedCameraUsecase* AdvancedCameraUsecase::Create(
    LogicalCameraInfo*              pCameraInfo,   ///< Camera info
    camera3_stream_configuration_t* pStreamConfig, ///< Stream configuration
    UsecaseId                       usecaseId)     ///< Identifier for usecase function
{
    CDKResult              result                 = CDKResultSuccess;
    AdvancedCameraUsecase* pAdvancedCameraUsecase = CHX_NEW AdvancedCameraUsecase;

    if ((NULL != pAdvancedCameraUsecase) && (NULL != pStreamConfig))
    {
        result = pAdvancedCameraUsecase->Initialize(pCameraInfo, pStreamConfig, usecaseId); //其中接著會調用CameraUsecaseBase::Initialize(m_pCallbacks),然后調用CameraUsecaseBase::CreatePipeline

        if (CDKResultSuccess != result)
        {
            pAdvancedCameraUsecase->Destroy(FALSE);
            pAdvancedCameraUsecase = NULL;
        }
    }
    else
    {
        result = CDKResultEFailed;
    }

    return pAdvancedCameraUsecase;
}

 至此,chi層根據APP的參數以及平臺、sensor的信息和topology xml的結構,選擇匹配的UsecaseID并創建了所需的pipeline。

 HFR配置流時的限制:

通過createConstrainedHighSpeedCaptureSession配置高速流
只能配置一個或者兩個流,一個預覽流,一個是拍照流
對預覽流的限制是usage為 GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_HW_COMPOSER | GRALLOC_USAGE_HW_RENDER
對錄像流的限制是usage為GRALLOC_USAGE_HW_VIDEO_ENCODER

、高通平臺如何獲取platform和camera sensor的capabilities?

  Snapdragon相機的setting界面中可以選擇Video quality和對應的Video High FrameRate,而選擇列表中的支持項是在camera服務啟動時根據平臺和camera sensor的輸出能力共同決定。
  在選擇一個Video quality后,HFR選項列表會被更新,其中的操作就是查詢當前分辨率支持的FPS,流程如下:

 (1)packagesappsSnapdragonCamerasrccomandroidcameraSettingsManager.java

   //查詢支持的fps并更新列表
  private void filterHFROptions() { ListPreference hfrPref = mPreferenceGroup.findPreference(KEY_VIDEO_HIGH_FRAME_RATE); if (hfrPref != null) { hfrPref.reloadInitialEntriesAndEntryValues(); if (filterUnsupportedOptions(hfrPref, getSupportedHighFrameRate())) { mFilteredKeys.add(hfrPref.getKey()); } } }
    private List<String> getSupportedHighFrameRate() {
        ArrayList<String> supported = new ArrayList<String>();
        supported.add("off");
        ListPreference videoQuality = mPreferenceGroup.findPreference(KEY_VIDEO_QUALITY);
        ListPreference videoEncoder = mPreferenceGroup.findPreference(KEY_VIDEO_ENCODER);
        if (videoQuality == null || videoEncoder == null) return supported;
        String videoSizeStr = videoQuality.getValue();
        int videoEncoderNum = SettingTranslation.getVideoEncoder(videoEncoder.getValue());
        VideoCapabilities videoCapabilities = null;
        boolean findVideoEncoder = false;
        if (videoSizeStr != null) {
            Size videoSize = parseSize(videoSizeStr);
            MediaCodecList allCodecs = new MediaCodecList(MediaCodecList.ALL_CODECS);
            for (MediaCodecInfo info : allCodecs.getCodecInfos()) {
                if (!info.isEncoder() || info.getName().contains("google")) continue;
                for (String type : info.getSupportedTypes()) {
                    if ((videoEncoderNum == MediaRecorder.VideoEncoder.MPEG_4_SP && type.equalsIgnoreCase(MediaFormat.MIMETYPE_VIDEO_MPEG4))
                            || (videoEncoderNum == MediaRecorder.VideoEncoder.H263 && type.equalsIgnoreCase(MediaFormat.MIMETYPE_VIDEO_H263))
                            || (videoEncoderNum == MediaRecorder.VideoEncoder.H264 && type.equalsIgnoreCase(MediaFormat.MIMETYPE_VIDEO_AVC))
                            || (videoEncoderNum == MediaRecorder.VideoEncoder.HEVC && type.equalsIgnoreCase(MediaFormat.MIMETYPE_VIDEO_HEVC))) {
                        CodecCapabilities codecCapabilities = info.getCapabilitiesForType(type);
                        videoCapabilities = codecCapabilities.getVideoCapabilities();
                        findVideoEncoder = true;
                        break;
                    }
                }
                if (findVideoEncoder) break;
            }

            try {
                // 獲取當前video size對應支持的fps
                Range[] range = getSupportedHighSpeedVideoFPSRange(mCameraId, videoSize);
                for (Range r : range) {
                    // To support HFR for both preview and recording,
                    // minmal FPS needs to be equal to maximum FPS
                    if ((int) r.getUpper() == (int) r.getLower()) {
                        if (videoCapabilities != null) {
                            if (videoCapabilities.areSizeAndRateSupported(
                                    videoSize.getWidth(), videoSize.getHeight(), (int) r.getUpper())) {
                                supported.add("hfr" + String.valueOf(r.getUpper()));
                                supported.add("hsr" + String.valueOf(r.getUpper()));
                            }
                        }
                    }
                }
            } catch (IllegalArgumentException ex) {
                Log.w(TAG, "HFR is not supported for this resolution " + ex);
            }
      .......
        }
        return supported;
    }

 (2) frameworksasecorejavaandroidhardwarecamera2paramsStreamConfigurationMap.java

    public Range<Integer>[] getHighSpeedVideoFpsRangesFor(Size size) {
        // 檢查當前選擇的video size是否支持HFR
        Integer fpsRangeCount = mHighSpeedVideoSizeMap.get(size);
        if (fpsRangeCount == null || fpsRangeCount == 0) {
            throw new IllegalArgumentException(String.format(
                    "Size %s does not support high speed video recording", size));
        }

        @SuppressWarnings("unchecked")
        Range<Integer>[] fpsRanges = new Range[fpsRangeCount];
        int i = 0;
        // 獲取當前video size支持的各fps
        for (HighSpeedVideoConfiguration config : mHighSpeedVideoConfigurations) {
            if (size.equals(config.getSize())) {
                fpsRanges[i++] = config.getFpsRange();
            }
        }
        return fpsRanges;
    }
其中 mHighSpeedVideoConfigurations 在下面接口中初始化 :frameworksasecorejavaandroidhardwarecamera2implCameraMetadataNative.java
    private StreamConfigurationMap getStreamConfigurationMap() {
        StreamConfiguration[] configurations = getBase(
                CameraCharacteristics.SCALER_AVAILABLE_STREAM_CONFIGURATIONS);
        StreamConfigurationDuration[] minFrameDurations = getBase(
                CameraCharacteristics.SCALER_AVAILABLE_MIN_FRAME_DURATIONS);
        StreamConfigurationDuration[] stallDurations = getBase(
                CameraCharacteristics.SCALER_AVAILABLE_STALL_DURATIONS);
        StreamConfiguration[] depthConfigurations = getBase(
                CameraCharacteristics.DEPTH_AVAILABLE_DEPTH_STREAM_CONFIGURATIONS);
        StreamConfigurationDuration[] depthMinFrameDurations = getBase(
                CameraCharacteristics.DEPTH_AVAILABLE_DEPTH_MIN_FRAME_DURATIONS);
        StreamConfigurationDuration[] depthStallDurations = getBase(
                CameraCharacteristics.DEPTH_AVAILABLE_DEPTH_STALL_DURATIONS);

     // 從camx層獲取high speed video的配置信息 HighSpeedVideoConfiguration[] highSpeedVideoConfigurations = getBase( CameraCharacteristics.CONTROL_AVAILABLE_HIGH_SPEED_VIDEO_CONFIGURATIONS);
ReprocessFormatsMap inputOutputFormatsMap = getBase( CameraCharacteristics.SCALER_AVAILABLE_INPUT_OUTPUT_FORMATS_MAP); int[] capabilities = getBase(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES); boolean listHighResolution = false; for (int capability : capabilities) { if (capability == CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES_BURST_CAPTURE) { listHighResolution = true; break; } }

     // 創建StreamConfigurationMap,檢查config信息 return new StreamConfigurationMap( configurations, minFrameDurations, stallDurations, depthConfigurations, depthMinFrameDurations, depthStallDurations, highSpeedVideoConfigurations, inputOutputFormatsMap, listHighResolution); }

(3) HAL層的 cameraInfo 就是本文第二部分分析ConfigureStreams時獲取的,在 vendorqcomproprietarycamxsrccorehalcamxhal3.cpp中調用pHALDevice->ConfigureStreams(pStreamConfigs);
   進入到了vendorqcomproprietarycamxsrccorehalcamxhaldevice.cpp:

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// HALDevice::ConfigureStreams
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
CamxResult HALDevice::ConfigureStreams(
    Camera3StreamConfig* pStreamConfigs)
{
    CamxResult result = CamxResultSuccess;

    // Validate the incoming stream configurations 
    result = CheckValidStreamConfig(pStreamConfigs); // 其中調用了 pHWEnvironment->GetCameraInfo(logicalCameraId, &cameraInfo);

    if ((StreamConfigModeConstrainedHighSpeed == pStreamConfigs->operationMode) ||
        (StreamConfigModeSuperSlowMotionFRC == pStreamConfigs->operationMode))
    {
        SearchNumBatchedFrames (pStreamConfigs, &m_usecaseNumBatchedFrames, &m_FPSValue);
        CAMX_ASSERT(m_usecaseNumBatchedFrames > 1);
    }
    else
    {
        // Not a HFR usecase batch frames value need to set to 1.
        m_usecaseNumBatchedFrames = 1;
    }

   ......
  return result; }

(4)platform、sensor的capabilities信息是在 vendorqcomproprietarycamxsrccorecamxhwenvironment.cpp 中初始化

VOID HwEnvironment::InitCaps()
{
......
if (CamxResultSuccess == result) {
     // 平臺、sensor的capabilities主要通過以下函數初始化 ProbeImageSensorModules(); //創建ImageSensorModuleDataManager,Initialize()時調用CreateAllSensorModuleSetManagers,其中會加載sensor的信息bin文件 EnumerateDevices(); InitializeSensorSubModules(); InitializeSensorStaticCaps(); //其中進一步調用ImageSensorModuleData::GetStaticCaps獲取sensor的capability result = m_staticEntryMethods.GetStaticCaps(&m_platformCaps[0]); // copy the static capacity to remaining sensor's for (UINT index = 1; index < m_numberSensors; index++) { Utils::Memcpy(&m_platformCaps[index], &m_platformCaps[0], sizeof(m_platformCaps[0])); } if (NULL != m_pOEMInterface->pInitializeExtendedPlatformStaticCaps) { m_pOEMInterface->pInitializeExtendedPlatformStaticCaps(&m_platformCaps[0], m_numberSensors); } } ...... }

(5) 在ProbeImageSensorModules中創建ImageSensorModuleDataManager時會加載模組的bin文件獲取sensor信息 :vendorqcomproprietarycamxsrccorecamximagesensormoduledatamanager.cpp

CamxResult ImageSensorModuleDataManager::CreateAllSensorModuleSetManagers()
{
    CamxResult                   result                  = CamxResultSuccess;
    ImageSensorModuleSetManager* pSensorModuleSetManager = NULL;
    UINT16                       fileCount               = 0;
    CHAR                         binaryFiles[MaxSensorModules][FILENAME_MAX];

    // 當前8150mtp使用的camera模組是ov12a10(wide),所以會加載com.qti.sensormodule.ofilm_ov12a10.bin
    fileCount = OsUtils::GetFilesFromPath(SensorModulesPath, FILENAME_MAX, &binaryFiles[0][0], "*", "sensormodule", "*", "bin");
    CAMX_ASSERT((fileCount != 0) && (fileCount < MaxSensorModules));

    m_numSensorModuleManagers = 0;

    if ((fileCount == 0) || (fileCount >= MaxSensorModules))
    {
        CAMX_LOG_ERROR(CamxLogGroupSensor, "Invalid fileCount", fileCount);
        result = CamxResultEFailed;
    }
    else
    {
        for (UINT i = 0; i < fileCount; i++)
        {
            result = GetSensorModuleManagerObj(&binaryFiles[i][0], &pSensorModuleSetManager);
            if (CamxResultSuccess == result)
            {
                m_pSensorModuleManagers[m_numSensorModuleManagers++] = pSensorModuleSetManager;
            }
            else
            {
                CAMX_LOG_ERROR(CamxLogGroupSensor,
                "GetSensorModuleManagerObj failed i: %d binFile: %s",
                i, &binaryFiles[i][0]);
            }
        }

        CAMX_ASSERT(m_numSensorModuleManagers > 0);
        if (0 == m_numSensorModuleManagers)
        {
            CAMX_LOG_ERROR(CamxLogGroupSensor, "Invalid number of sensor module managers");
            result = CamxResultEFailed;
        }
    }

    return result;
}

 獲取sensor static capability的調用時序圖:

  

 上面com.qti.sensormodule.ofilm_ov12a10.bin可以修改對應xml編譯更新,路徑在:vendorqcomproprietarychi-cdkvendorsensordefaultov12a10

  

 從ov12a10_sensor.xml可以看到1080p支持最大60fps:

  

  需要注意的是,修改xml參數frameRate為120,更新.bin后app的設置中的確會增加120 fps選項,但sensor的輸出能力如果只能達到1080p@60fps的話,錄制結果會卡頓,由于sensor的輸出幀率低于編碼率,所以插入了很多重復幀。

  Ps: HFR Usecase需要幀率大于等于120fps。

總結

以上是生活随笔為你收集整理的Android:高通平台Camera HFR Usecase分析的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

久久成人欧美 | 成人在线观看免费视频 | 一区二区三区四区久久 | 成人国产精品 | 在线岛国av | 免费色视频网站 | 在线成人中文字幕 | 有码中文字幕 | 国产成人精品一区二区三区福利 | 日韩免费在线看 | 午夜成人免费影院 | 伊人天天狠天天添日日拍 | 日韩综合第一页 | 开心色婷婷| 欧美日韩国产一区二 | 久久免费美女视频 | 午夜久久视频 | 一区免费观看 | 国产123区在线观看 国产精品麻豆91 | 91激情 | 国产精品一区二区精品视频免费看 | 久久er99热精品一区二区 | 国产在线观看二区 | 亚洲视频久久 | 免费h精品视频在线播放 | 欧美怡红院 | 97偷拍视频 | 91亚洲精品久久久蜜桃 | 色婷婷综合在线 | 日日干激情五月 | 日本一区二区免费在线观看 | 99re视频在线观看 | 亚洲免费资源 | 久草在线网址 | 国产99久久精品 | 欧美日韩亚洲在线 | 日本 在线 视频 中文 有码 | av成人动漫在线观看 | 91成人免费看 | 二区三区在线 | 国产三级精品三级在线观看 | 免费亚洲视频 | 免费在线播放av电影 | 久草在线视频免赞 | 成人黄色短片 | 国产一级淫片免费看 | 国产精品剧情 | 亚洲精品456在线播放乱码 | 国产在线小视频 | 日本精品视频在线播放 | 色国产在线 | 日韩欧美一区二区三区免费观看 | 日韩两性视频 | 欧美精品亚州精品 | 一级a性色生活片久久毛片波多野 | 亚洲国产精品推荐 | 91av视频免费观看 | 97国产精品视频 | 丰满少妇久久久 | 国产精品久久久久婷婷二区次 | 成人黄色影片在线 | 日日爽天天爽 | 夜色成人av | 久草在在线 | 亚洲成人精品 | 久久精品综合一区 | 在线观看亚洲视频 | 六月丁香六月婷婷 | 国产不卡一 | 在线观看中文字幕网站 | 精品视频成人 | 亚洲欧洲av在线 | 中文字幕在线观看你懂的 | 午夜在线观看影院 | 久久99久久99精品 | 五月天亚洲激情 | 婷婷久久网站 | 久久96国产精品久久99漫画 | 亚洲 在线 | aa一级片 | 在线中文字幕电影 | 亚洲视频播放 | 亚洲国产成人在线观看 | 国产黄a三级 | 国内99视频 | 精品久久九九 | 成人超碰在线 | 色综合久久久久久久久五月 | 97精品国产97久久久久久久久久久久 | 97视频成人| 黄色av一级片 | 天海翼一区二区三区免费 | 欧美成年人在线视频 | 黄色成人av | 久久久久国产成人免费精品免费 | 日韩欧美高清一区二区 | 久久狠狠干 | 天天操天天操天天干 | a国产精品 | 国产99免费视频 | 国产精品久久影院 | 久久美女免费视频 | 黄色免费视频在线观看 | 久久综合九色99 | 91九色国产视频 | 日韩精品黄 | 欧美黑人性猛交 | 久久久影院一区二区三区 | 色婷久久 | 久久中国精品 | 丁香激情综合久久伊人久久 | 不卡视频在线看 | 国内精品久久久久久久影视麻豆 | 国产一级视频在线免费观看 | 欧美日韩网站 | 91精品国产乱码在线观看 | 美女视频永久黄网站免费观看国产 | av三级在线免费观看 | 五月天狠狠操 | 中文字幕av全部资源www中文字幕在线观看 | 日本精品在线视频 | 日批视频在线 | 欧美一级视频免费 | 99久久婷婷国产综合亚洲 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 伊人久久国产精品 | 欧美射射射 | 999男人的天堂 | 欧美在线观看小视频 | 久久精品一区二区三区视频 | 日韩欧美国产免费播放 | 天堂中文在线视频 | 亚洲成人精品在线 | 四虎影视成人精品国库在线观看 | 日韩精品免费一区 | 欧美精品一区二区免费 | 久久综合五月天婷婷伊人 | 国产91精品在线播放 | 国产免费叼嘿网站免费 | 欧美视频在线观看免费网址 | 久久成人午夜视频 | 丁香婷婷综合五月 | 热久久电影 | 国产精品久久久99 | 欧美日产在线观看 | 日韩久久一区 | 91精品久久久久久久91蜜桃 | 亚洲桃花综合 | 国产精品白浆视频 | 久久观看最新视频 | 国产在线97 | 91伊人影院 | 精品国产_亚洲人成在线 | 日韩三级视频在线看 | 欧美孕妇与黑人孕交 | japanesefreesex中国少妇 | 免费成人在线视频网站 | 久久人人爽人人爽人人片av免费 | 好看av在线 | 国产精品黄色在线观看 | 中文字幕日韩国产 | 国产免费国产 | 91成人欧美 | 久久精品国产精品亚洲精品 | 久久爱综合 | 日韩在线无| 日日夜夜免费精品视频 | 日韩免费精品 | 国产综合在线观看视频 | 91chinese在线| 免费三级骚 | 成人午夜电影在线观看 | 亚洲精品国产精品乱码不99热 | 久久天天综合网 | av福利网址导航 | 亚洲人成综合 | 亚洲综合成人在线 | 久久久国产精品麻豆 | av片在线观看 | 在线观看国产日韩欧美 | 亚洲精品五月天 | 亚洲专区 国产精品 | 国产一区二区三区免费视频 | 在线观看视频一区二区三区 | 国产九九精品视频 | 中文字幕日本在线观看 | 国产九九九视频 | 午夜视频福利 | 色偷偷网站视频 | 亚洲最快最全在线视频 | 人人爽人人香蕉 | 九色精品 | 午夜精品久久一牛影视 | 久久99操| 97视频免费观看 | 中文字幕黄色av | 久久夜色精品国产欧美一区麻豆 | 91亚色视频在线观看 | 久久综合五月天婷婷伊人 | 欧美日韩不卡在线 | 国产一区欧美一区 | 激情五月婷婷网 | 日韩理论电影在线 | 天天爱天天射 | 96国产精品视频 | 99亚洲精品 | 九九九九精品九九九九 | 国产精品99免费看 | 色播五月激情综合网 | 毛片精品免费在线观看 | 五月天国产精品 | 国产精品久久久久久久久久不蜜月 | 男女激情免费网站 | 日韩视频一区二区在线观看 | 日韩在线二区 | 女女av在线 | 福利电影久久 | 狠狠色伊人亚洲综合网站色 | 国产视频一区二区三区在线 | 日韩欧美在线免费观看 | 91麻豆精品国产自产在线游戏 | 超碰成人av | 国产96在线观看 | 久久 在线 | 97视频在线观看网址 | 国产一二三区av | 午夜精品一区二区三区免费视频 | 国产69精品久久久久久久久久 | 亚洲欧洲日韩 | 国产在线观看免费观看 | 亚洲综合涩 | 久久精选视频 | 91精品在线免费 | 中文字幕一区二区三区久久 | 亚洲va综合va国产va中文 | 成 人 黄 色 视频 免费观看 | 亚洲一级二级三级 | 色五月成人 | av最新资源 | 午夜私人影院 | 色婷婷电影网 | 91亚洲精品久久久 | 精品视频123区在线观看 | 探花视频免费观看 | 久久综合狠狠综合久久激情 | 国产精品久久影院 | 亚洲国产手机在线 | 国产精品 国产精品 | 天天综合人人 | 九九热精品视频在线播放 | 久久精品亚洲精品国产欧美 | 欧美午夜性 | 亚洲国产免费av | 亚洲天堂精品视频 | 国产日韩精品一区二区在线观看播放 | 中文字幕在线字幕中文 | 日韩69视频 | 欧美乱熟臀69xxxxxx | 欧美在线视频一区二区三区 | 婷婷久月| 国产一区二区高清 | 丁香色婷婷 | 中文字幕乱码在线播放 | 在线观看网站你懂的 | 亚洲1区在线 | 久久久精品视频成人 | 天天干婷婷 | 人人藻人人澡人人爽 | 精品一区二三区 | 久久黄色片子 | 中文字幕免费高清在线观看 | 精品一区二区免费视频 | 中文字幕乱码电影 | 日韩免费一区 | 色91在线| 国产精品久久久久久欧美 | 免费在线观看成人av | 91亚洲在线观看 | 久久精品视频在线免费观看 | 天天干天天搞天天射 | 美女黄濒 | 亚洲精品五月天 | 免费看一级黄色大全 | 色婷婷激情四射 | 黄色资源网站 | 国产色综合 | 久久欧美精品 | 亚洲天堂网在线播放 | 91一区在线观看 | 三级av网| 久久精品人人做人人综合老师 | 深夜激情影院 | 欧美成人在线免费 | 2024av| 日韩在线观看不卡 | 一区二区三区在线视频观看58 | 蜜桃视频精品 | 亚洲一区动漫 | 婷婷色在线播放 | 激情久久久久久久久久久久久久久久 | 96av在线视频| 九热在线 | 99视频精品全部免费 在线 | 国产综合91 | 美女av电影 | 久久一区国产 | 国产精品第一页在线 | 日韩在线视频线视频免费网站 | 国产精品一区二区在线播放 | 亚洲综合涩 | 亚洲黄色在线 | 欧美日韩在线电影 | 少妇bbbb搡bbbb搡bbbb | 日韩久久精品一区二区三区 | 日韩69av | 天天操天天弄 | 久久精品一区二区国产 | 日韩免费成人av | 国产不卡在线看 | 久久久久亚洲最大xxxx | 婷婷视频导航 | 黄色精品一区二区 | 免费成视频| 国产成视频在线观看 | 欧美精品资源 | 免费观看黄色12片一级视频 | 国产中文字幕在线观看 | 欧美精品一区二区蜜臀亚洲 | 久久久久免费精品 | 亚洲国产激情 | 亚洲成人精品影院 | 国产一级一级国产 | 久久综合操 | 国产成人久久久77777 | 亚洲国产人午在线一二区 | 国产精品麻| 午夜精品av在线 | 欧美日韩精 | 亚洲国产成人高清精品 | 国内精自线一二区永久 | 久久国内视频 | 久色伊人 | 久久国产综合视频 | 综合激情网 | 日韩午夜在线 | 天天拍天天色 | 欧美一区二区三区不卡 | a在线播放 | 超碰国产在线 | 中文国产在线观看 | 在线观看精品一区 | 色综合久久五月 | 日日夜夜亚洲 | 正在播放国产一区二区 | 欧美一级视频在线观看 | 一区二区亚洲精品 | 色吧av色av| 精品国产自在精品国产精野外直播 | 中文字幕888 | 在线香蕉视频 | 中文字幕一区二区三区四区在线视频 | 日韩网站免费观看 | 美女黄色网在线播放 | 日本久久成人 | 天堂久久电影网 | 丁香激情五月 | 精品国产欧美一区二区 | 亚洲精品国产自产拍在线观看 | 亚洲国产高清视频 | 国产午夜小视频 | 一区二区三区四区五区在线视频 | 日韩在线视频免费观看 | 最近字幕在线观看第一季 | 免费看国产曰批40分钟 | 午夜精品99久久免费 | 中文字幕黄色av | se婷婷| 婷婷色中文字幕 | 在线看的毛片 | 久久你懂的 | 欧美日韩在线视频一区二区 | 3d黄动漫免费看 | 国色天香在线观看 | 麻豆91在线看 | 精品国产一二三四区 | 777xxx欧美| 午夜免费福利视频 | 日韩免费视频观看 | 婷婷中文在线 | 欧美巨乳波霸 | 最近久乱中文字幕 | 99久久99视频只有精品 | 国产69精品久久99不卡的观看体验 | 五月天婷亚洲天综合网鲁鲁鲁 | 欧美一级片在线免费观看 | 天天爱av导航| 欧美一区二区在线免费观看 | 成人毛片一区 | www.国产在线视频 | 91亚洲欧美激情 | 日韩免费av网址 | 亚洲综合色丁香婷婷六月图片 | 亚洲精品一区二区精华 | 精品久久久久久久久久岛国gif | 天天操夜夜做 | 久久久久久高潮国产精品视 | 色天天综合久久久久综合片 | www操操操 | 天天干天天射天天插 | 中文字幕 国产 一区 | 色综合久久久久久中文网 | 久久综合九色综合欧美就去吻 | 国产精品毛片一区视频播不卡 | 成人免费视频网址 | 精品久久精品久久 | 夜夜躁日日躁狠狠久久88av | 精久久久久 | 五月婷婷av | 最近日本mv字幕免费观看 | 国产精品99久久免费观看 | 久久精品国产亚洲精品 | 久久草在线免费 | 国产在线久久久 | 国产精品国产亚洲精品看不卡 | 国产精品久久久久久久免费大片 | 国产高清在线观看 | 久久99欧美 | 国产精品久久久久久69 | 久草在线免费电影 | 精品视频999 | 国产麻豆精品一区二区 | 一区二区三区动漫 | 中文字幕在线观看第一区 | 日日爽天天操 | 国产视频 久久久 | 精品播放 | 久久9999久久 | 免费黄色看片 | 在线视频 区| 成人黄性视频 | 麻豆视频在线观看 | 97视频久久久 | 干干日日| 伊人天天综合 | 五月婷婷六月综合 | 欧美一性一交一乱 | 成人影片在线免费观看 | 欧美视频xxx | 久久久久伦理电影 | 97超碰在线久草超碰在线观看 | 黄色小网站免费看 | www免费 | 精品一区二区在线观看 | 精品国产一区二区三区噜噜噜 | 国产一区二区在线免费视频 | 日韩免费一级a毛片在线播放一级 | 最近中文字幕免费 | 日韩理论在线 | 天天操伊人 | 二区中文字幕 | 久久婷亚洲五月一区天天躁 | 中文在线a在线 | 中文字幕一区在线观看视频 | 国产精品久久久久免费 | 久久成人国产精品入口 | 在线观看中文字幕dvd播放 | 久久都是精品 | 超级碰99 | 中文字幕日韩精品有码视频 | 久久久久国产a免费观看rela | 国产精品久久久久久久久久ktv | 在线视频中文字幕一区 | 人人爽人人干 | 国产精品黄 | 欧美午夜精品久久久久 | 中文字幕av在线免费 | 在线国产一区二区三区 | 国产成人av在线影院 | 91秒拍国产福利一区 | 亚洲成人一区 | 在线视频 你懂得 | 国产免费观看高清完整版 | 国产亚洲精品综合一区91 | 免费合欢视频成人app | 狠狠色丁香久久婷婷综合_中 | 久久久久国产精品一区二区 | 九草在线视频 | 日韩av中文在线观看 | 久久久久久久久爱 | 亚洲精品乱码久久久久久蜜桃动漫 | 亚洲国产剧情 | 久久久精品国产免费观看同学 | 国语精品免费视频 | 超碰夜夜 | 国产精品久久久久久久毛片 | 久久国产精品精品国产色婷婷 | 国产精品理论视频 | 日韩在线观看一区二区 | 日日操操操 | 亚洲成人999 | 国产高清av在线播放 | 日韩欧美在线第一页 | 99精品亚洲 | 综合网中文字幕 | 日本巨乳在线 | 奇米导航 | 久久免费视频在线 | 天天干,夜夜爽 | 国产xx视频| 欧美一级性 | 三级av网站 | 91麻豆文化传媒在线观看 | 91成人免费看片 | 亚洲永久精品一区 | 国产精品18久久久久久久 | 免费在线一区二区三区 | 婷婷六月丁香激情 | 免费网站v| 欧美日韩精品综合 | 成年人在线看视频 | 成年人天堂com | 婷婷综合av | 国产精品系列在线 | 91九色视频网站 | a级国产乱理论片在线观看 特级毛片在线观看 | 99久久99久久精品国产片 | 欧美激情精品久久久久久变态 | 久久视频二区 | 欧美精品久久久久久久久免 | 手机成人av| 国产精品一区久久久久 | 国产免费av一区二区三区 | 麻豆mv在线观看 | 国产精品一区二区免费在线观看 | 国产一级久久久 | 午夜三级理论 | 可以免费看av | 国产免费久久精品 | 有码一区二区三区 | 麻豆国产在线播放 | 麻豆免费视频 | 91亚瑟视频| 亚洲精品在线观 | 麻豆视频在线看 | 亚洲精品999 | 国产一级精品在线观看 | 天天爽夜夜爽精品视频婷婷 | 六月婷婷久香在线视频 | 99国产精品免费网站 | 国产精品12345 | 国产高清区 | 色瓜| 中文字幕中文字幕在线中文字幕三区 | 免费中文字幕视频 | 狠狠狠色狠狠色综合 | 中文区中文字幕免费看 | 亚洲精品在线免费看 | 激情综合五月 | 五月天丁香亚洲 | 超碰成人av | 久爱精品在线 | 久久这里只有精品视频首页 | 欧美日韩精品久久久 | 欧美va天堂va视频va在线 | 五月婷婷在线视频 | 99热最新精品 | 成人cosplay福利网站 | 国产在线欧美日韩 | 久久视频在线 | 国产 日韩 在线 亚洲 字幕 中文 | 一区二区三区免费在线观看 | 中文字幕在线观看网站 | 婷婷丁香六月 | 日韩极品在线 | 国产精品久久99综合免费观看尤物 | 婷婷亚洲五月色综合 | 成年一级片 | 在线午夜电影神马影院 | 美女久久视频 | 国产精品一区二区在线看 | 国产亚洲精品福利 | 国产小视频在线免费观看视频 | 最近最新中文字幕 | 在线观看视频一区二区三区 | 久久视频99| 一区二区精品在线观看 | 久久热亚洲 | 国内三级在线观看 | 欧美日韩免费观看一区=区三区 | 国产一区二区三精品久久久无广告 | 成人免费共享视频 | 国产专区第一页 | 视频国产一区二区三区 | 中文字幕一区二 | 国产色网站 | 最新婷婷色 | 在线免费观看国产视频 | 久久性生活片 | 久久综合久久综合这里只有精品 | 天天做天天爱天天综合网 | 97超碰免费| 日韩精品视频在线观看网址 | 亚洲精品黄色 | 丁香激情五月 | 免费日韩av电影 | 免费在线播放黄色 | 一区二区三区在线免费观看视频 | 国产资源中文字幕 | 欧美成人视 | 视频成人 | 日韩在线中文字幕视频 | 欧美欧美 | 欧美日韩精品影院 | 99久久婷婷国产精品综合 | 婷婷六月综合亚洲 | 97网在线观看 | 天天操 夜夜操 | 精品视频在线播放 | 手机看片99| 欧美黄色免费 | 日本在线观看视频一区 | 久久久精品网站 | 国产在线a免费观看 | 国产区精品区 | 国产在线黄色 | 福利电影一区二区 | 在线精品视频免费播放 | 在线看片中文字幕 | 午夜精品视频福利 | 中文日韩在线视频 | 午夜视频在线观看一区二区三区 | 免费高清影视 | 亚洲成av人片在线观看 | 91色视频| 西西www4444大胆在线 | 成人av动漫在线观看 | 欧美成人高清 | 亚洲免费观看在线视频 | 在线国产中文 | 久久99婷婷| 69久久久 | 欧美精品色 | 日本色小说视频 | 日韩在线观看你懂得 | 手机在线日韩视频 | 精品久久综合 | 亚洲一区二区三区精品在线观看 | 国产丝袜美腿在线 | 五月香视频在线观看 | 久久涩视频 | 婷婷免费视频 | 高清在线一区二区 | 狠狠色丁香九九婷婷综合五月 | 香蕉影院在线播放 | 91精选在线观看 | 久久这里有精品 | 久久久久国产a免费观看rela | 国产日韩欧美在线影视 | 国产精品福利在线 | 一区三区在线欧 | 国产在线精品国自产拍影院 | 91最新中文字幕 | 久久超碰在线 | 99精品视频免费在线观看 | 波多野结衣久久精品 | 黄色av电影一级片 | 久99久中文字幕在线 | 国产精品 日韩精品 | 国产高清小视频 | 亚洲国产黄色片 | 九九99视频| 亚洲国产精品va在线看黑人 | 欧美另类交人妖 | 精品国产一区二区三区久久影院 | 免费福利视频网 | 久久国语| 91精品国产91p65 | 一区二区三区中文字幕在线观看 | 国产福利精品在线观看 | 亚洲精品在线观看视频 | 免费亚洲视频在线观看 | 99在线热播 | 亚洲成人频道 | 免费91麻豆精品国产自产在线观看 | 99re热精品视频 | av一级久久 | 国产精品久久一卡二卡 | 99精品一区二区 | 国产精品免费不卡 | 在线а√天堂中文官网 | 久久久久久久久久电影 | 正在播放一区 | 日日干干夜夜 | 欧美日韩精品在线一区二区 | 亚洲毛片久久 | 亚洲作爱视频 | 国产精品黄色 | 日韩欧美成人网 | 国产1级毛片 | 日本精品一区二区在线观看 | 日韩在线 一区二区 | 91av99| 久久精品国产亚洲 | 黄色免费观看视频 | 亚洲无吗天堂 | 天天摸日日摸人人看 | 久久综合色8888 | 色福利网站 | 天天天天爱天天躁 | 亚洲欧美视频一区二区三区 | 久久美女精品 | 三级免费黄 | 最近免费中文视频 | av资源在线看 | 一区三区视频在线观看 | 亚洲成a人片77777kkkk1在线观看 | 国产午夜三级一区二区三桃花影视 | 高清不卡毛片 | 午夜精品福利一区二区三区蜜桃 | 99精品国产成人一区二区 | 午夜91视频 | 天天曰 | 精品亚洲欧美一区 | 国产午夜免费视频 | 精品国产免费一区二区三区五区 | 97爱爱爱 | 欧美国产亚洲精品久久久8v | 欧美精品做受xxx性少妇 | 中文字幕av有码 | 五月天六月丁香 | 日韩欧美在线高清 | 久久精品官网 | 婷婷丁香在线 | 欧美a级在线免费观看 | 免费视频成人 | 国产精品一区二区视频 | av大全在线免费观看 | www色网站 | 日韩一区二区三区在线观看 | 亚洲天堂自拍视频 | 日韩欧美电影在线观看 | 99精品在线免费 | 国产成人333kkk| 精品婷婷 | 午夜婷婷在线播放 | 久久久久久免费视频 | 免费观看黄 | 久久久五月天 | 在线观看午夜 | 人人插人人插 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 在线观看视频福利 | 成人a级网站| 国产三级久久久 | 免费色av| 久久久久五月天 | 精品国产电影 | 国产专区第一页 | 亚洲视频 中文字幕 | 新av在线 | 天天爽天天碰狠狠添 | 丁香影院在线 | 精品国产一区二区三区久久影院 | av在线播放网址 | 国产精品福利久久久 | 久久尤物电影视频在线观看 | 天天射射天天 | 亚洲国产精品电影 | 成人超碰97| 一本色道久久综合亚洲二区三区 | 毛片.com| 五月婷激情 | 国产a国产a国产a | 久草精品在线观看 | 中文字幕 国产视频 | 99c视频在线 | 国产999精品久久久久久绿帽 | 中文字幕亚洲高清 | 日本一区二区免费在线观看 | 日韩欧美一区二区在线播放 | 成人小电影在线看 | 丁香资源影视免费观看 | 国产视频在线观看免费 | 国产 在线观看 | 久久久久久久久久网 | 国产精品美女久久久免费 | 国产高清在线不卡 | 免费视频成人 | 久久综合一本 | 免费观看国产精品 | 福利一区二区三区四区 | 久久久香蕉视频 | 黄色avwww| 色中文字幕在线观看 | 伊人国产视频 | 天天婷婷| av天天色| 欧美在线1区 | sesese图片 | 超碰九九 | 久久精品精品电影网 | 国产99在线| 99在线精品视频 | 天天综合狠狠精品 | 在线免费观看黄 | www婷婷 | 91av亚洲| 一级α片免费看 | 成人aⅴ视频| 亚洲欧洲av | 欧美日韩国产伦理 | 国产精品一二 | 99草在线视频| 综合色影院 | 免费观看的av | 国产一区二区三区网站 | 久久a级片| 日韩最新av| 一区二区三区中文字幕在线 | 久久久高清一区二区三区 | 亚洲最大成人免费网站 | 麻花天美星空视频 | 午夜丰满寂寞少妇精品 | 国产在线精品福利 | 欧美日本在线观看视频 | 中文字幕观看av | 日韩videos| 久久综合九色综合97_ 久久久 | 99久久www | 日韩字幕在线 | 人人玩人人添人人 | 五月婷婷av在线 | 国产精品av久久久久久无 | 狠狠88综合久久久久综合网 | 综合网天天 | 久久好看免费视频 | 久久久激情网 | 国产在线中文字幕 | 亚洲国产成人在线播放 | 国产一级免费观看视频 | 麻豆传媒在线免费看 | 日韩一级成人av | 亚洲v欧美v国产v在线观看 | 免费看毛片网站 | 国产婷婷一区二区 | 亚洲电影成人 | 久久超碰99 | 亚洲高清91 | 蜜桃视频在线观看一区 | 亚洲精品久久激情国产片 | www.天天操.com | 国产va在线| 久久久久久久99精品免费观看 | aaawww| 欧美视频二区 | 亚洲黄色成人 | av视屏在线播放 | 四虎国产精品永久在线国在线 | www日韩视频 | 欧美aa级 | 精品无人国产偷自产在线 | 四虎亚洲精品 | 免费日韩 精品中文字幕视频在线 | 国产精品系列在线观看 | av不卡中文字幕 | 成人免费在线观看电影 | avav片| 日本中文字幕高清 | 深爱婷婷激情 | 国产精久久久久久妇女av | 久久综合九色 | 国产精品理论片在线播放 | 天天摸日日操 | 久久久久久影视 | 国产精品露脸在线 | 性色av一区二区三区在线观看 | 亚洲精品女 | 色噜噜在线观看视频 | 最近av在线| 亚洲精品国产电影 | 午夜精品久久一牛影视 | 日韩av资源在线观看 | 日韩精品在线观看av | 激情欧美丁香 | 91精品国产综合久久福利不卡 | 亚洲国产日本 | 中文av字幕在线观看 | 精品理论片 | 欧美视频国产视频 | 又污又黄的网站 | 激情图片久久 | 国产日韩欧美在线播放 | 日本爱爱片| 亚洲国产中文字幕在线 | 亚洲专区一二三 | 国产在线观看av | 国产精品6| 狠狠色丁香婷综合久久 | 欧美精品乱码久久久久 | 久久刺激视频 | 91天天视频| www.色五月.com | 激情综合啪啪 | 欧美精品天堂 | 精品91视频 | 亚洲成人频道 | 久99久精品 | 在线观看视频一区二区 | 成人国产网址 | 在线网站黄 | 国产精品久久久久久五月尺 | 久久麻豆视频 | av一区二区三区在线观看 | 中文字幕乱在线伦视频中文字幕乱码在线 | 人人插人人舔 | 久草网免费 | 日韩av电影免费观看 | 国产黄色片免费看 | 国产亚洲精品久久久久久 | 欧美精品一区二区免费 | 我要看黄色一级片 | 国产91综合一区在线观看 | 国产真实精品久久二三区 | 中文字幕888| 国产一级二级在线播放 | 色婷婷一区 | 国内揄拍国内精品 | 日韩在线免费播放 | 97人人模人人爽人人少妇 | www.色婷婷 | 中文字幕有码在线观看 | 在线免费视频a | 国产精品门事件 | 国产日韩精品欧美 | 久久人网 | 一区二区中文字幕在线 | 日本久久精 | 日日天天av | 天天爽天天爽夜夜爽 | 中文字幕你懂的 | 色在线网站 | 99精品福利 | 97在线观看 | 成年人毛片在线观看 | 亚洲一级免费电影 | 91精品国产91久久久久 | 在线观看中文字幕av | 久操97 | 在线观看国产日韩欧美 | 在线观看网站av | 成人香蕉视频 | 美女网站在线免费观看 | 国产精品综合久久久久 | www.久久久com | 国产精品久久久久久久久久新婚 | 免费av网址在线观看 | 日韩精品综合在线 | 亚洲在线观看av | 狠狠综合久久av | 夜夜夜夜操 | 国产精品久久中文字幕 | 五月婷婷精品 | 亚洲高清视频在线观看 | 中文字幕在线观 | 成 人 黄 色视频免费播放 | 伊人婷婷在线 | 国产小视频免费在线观看 | 久草在线视频精品 | 在线国产视频 | 久久久精品欧美一区二区免费 | 亚洲免费在线播放视频 | 欧美日韩免费在线视频 | 精品视频免费久久久看 | 久久蜜臀av | 色亚洲网 | 美女网站视频免费都是黄 | 亚洲精品视频免费观看 | 免费在线播放av电影 | 欧美日韩高清不卡 | 国产精品美女久久久免费 | 麻豆高清免费国产一区 | 97国产在线 | 视频一区二区国产 | 国产小视频国产精品 | 黄色高清视频在线观看 | 日韩无在线 | 五月婷婷天堂 | 欧美精品v国产精品v日韩精品 | 五月婷婷免费 | 欧美精品久久久久久久亚洲调教 | 九九在线播放 | 伊人国产在线观看 | 成人在线观看资源 | 精品一区精品二区 | 五月婷婷激情综合网 | 久久av高清| 中文字幕乱码电影 | 中文字幕亚洲综合久久五月天色无吗'' |