Lifelong Learning with you, this is Android programmer

This article covers some of the basics of Android development. Here’s what you’ll learn:

1. Overview 2. Basic component concepts 3

A, an overview of

Reviewing the history of Qualcomm platform Camera HAL, Qualcomm used QCamera & MM-Camera architecture before, but in order to better control the underlying hardware (Sensor/ISP and other key hardware) and facilitate mobile phone manufacturers to customize some functions, camX-Chi architecture is proposed now. Because in the CamX – CHI couldn’t see the shadow of the old architecture before, so it is a completely new architecture, it will be some highly unified functional interface pulled out into the CamX, parts can be customized in CHI modified for different manufacturers, to realize the features of their own, the benefits of this design is obvious, Even if developers don’t know much about CamX, they can easily add custom features, lowering the threshold for developers to develop on Qualcomm’s platform.

Let’s start with the most intuitive directory structure to do a simple understanding of the architecture, the following is the camX-Chi basic directory structure:

This part of the code is located in vendor/qcom/proprietary/ :

Among them, CAMX represents the code implementation set (CAMX) of general functional interface, and Chi-CDK represents the code implementation set (CHI) of customizable requirements. It can be seen from the figure that camX part is the implementation of HAL3 interface on the top and on the bottom

It communicates with Kernel through V4L2 framework, and keeps interaction with CHI through mutual Dlopen SO library and acquisition of each other’s operation interface.

Camx/has the following main directories:

  • Core / : The core implementation module for CAMX, which also contains the Hal/directory primarily used to implement the HAL3 interface and the CHI/directory responsible for interacting with CHI
  • CSL / : It is used to store the communication module mainly responsible for CAMX and Camera driver, and provides a unified camera driver control interface for CAMX
  • HWL /: stores hardware nodes with independent computing capabilities. These nodes are managed by CSL
  • SWL /: Used to store nodes that do not have independent computing power and must be implemented by CPU

Chi-cdk/has the following main directories:

  • Chioverride /: The core module that houses the CHI implementation and is responsible for interacting with CAMX and implementing CHI’s overall framework and specific business processes.
  • Bin /: stores platform-related configuration items
  • Topology /: Used to store user-customized Usecase XML configuration files
  • Node /: Node for storing user-defined functions
  • Module /: Stores configuration files of different sensors. This part is required when initializing a sensor
  • Tuning /: Used to store configuration files for effect parameters in different scenarios
  • Sensor /: Stores the private information and register configuration parameters of different sensors
  • Actuator /: Stores configuration information of different focusing modules
  • Ois / : Used to store the configuration information of the anti-shake module
  • Flash / : Stores the configuration information of the flash module
  • Eeprom /: Stores the configuration information of the EEPROM external storage module
  • Fd /: Stores the configuration information of the face recognition module

2. Basic component concepts

1. Usecase

As the largest abstraction in CAMX-CHI, it contains several pipelines for specific functions, which are implemented in CHI through the Usecase class, which is responsible for business processing and resource management.

The Usecase class, which provides a set of common interfaces, serves as the base class for all existing usecases, among which, AdvancedCameraUsecase inherits from CameraUsecaseBase, Most scenes in a camera are instantiated through AdvancedCameraUsecase, which includes several main interfaces:

  • Create(): This method is static and is used to Create an instance of AdvancedCameraUsecase that gets the corresponding Usecase configuration information in the XML in its constructor.
  • ExecuteCaptureRequest(): This method is used to send a Request.
  • ProcessResultCb(): This method is registered as a callback method during Session creation and is called to send the results to AdvancedCameraUsecase once Session data processing is complete.
  • ProcessDriverPartialCaptureResult(): This method is registered as a callback method during Session creation. When partial meta data is generated in the Session, this method is called to send it to the AdvancedCameraUsecase.
  • ProcessMessageCb(): This method is registered as a callback method during Session creation and is called to AdvancedCameraUsecase whenever any event is generated in the Session.
  • ExecuteFlush(): This method is used to flush AdvancedCameraUsecase.
  • Destroy(): This method is used to safely Destroy AdvancedCameraUsecase.

The customizable part of Usecase is abstracted into the common_usecase.xml file. Here are a few of the main tag meanings: Usecase

  • UsecaseName: Represents the name of the Usecase. The definition of the Usecase will be found based on this name.
  • Targets: indicates the collection of data streams used for output, including the format of data streams and the range of output Size.
  • Pipelines: Used to define all pipelines that the Usecase can be used, at least one of which must be defined.

2. Feature

It represents a specific function, which requires the combination of multiple pipelines and is unified managed by Usecase. It is realized in CHI through Feature class, and there is no corresponding definition in XML. Specific Feature selection is completed in Usecase. During Feature creation, Usecase and Usecase can access each other’s resources by passing in an instance of Usecase.

The following are the existing features, in which Feature exists as a base class and defines a series of common methods.

Several common features:

  • FeatureHDR: An HDR feature that manages one or more pipeline resources and their flows to produce HDR images.
  • FeatureMFNR: The MFNR functionality is divided internally into several large processes, including Prefiltering, Blending, Postfilter, and finally OfflineNoiseReproces(this one is optionally enabled), each with its own pipeline.
  • FeatureASD: For the realization of AI functions, during the preview, receive each frame of data, analyze the AI recognition output of the current scene, and send it to the upper layer through methods such as metadata for subsequent processing.

3. Session

Abstract control unit used to manage pipeline, a Session has at least one Pipeine, and controls all hardware resources, controls the flow of each internal pipeline request and data input and output, it has no customizable part, So the Session is not defined as a separate unit in the XML file in CHI.

Session is realized mainly through the Session class in CamX, whose main interfaces are as follows:

  • Initialize(): Performs Session initialization based on the passed SessionCreateData parameter.
  • NotifyResult(): The internal Pipeline sends results to the Session through this interface.
  • ProcessCaptureRequest(): This method is called when the user decides to send a Request into the Session.
  • StreamOn(): Begins the hardware data transfer through the passed Pipeline handle.
  • StreamOff(): Stops hardware data transfer through the passed Pipeline handle.

4. Pipeline

As a collection of all resources providing a single specific function, it maintains all hardware resources and data flow. Each Pipeline includes Node/Link, which is realized through Pipeline class in CamX. It is responsible for the maintenance of software and hardware resources of the whole Pipeline and the processing of business logic. Let’s take a quick look at the main interfaces of this class:

  • Create(): This method is a static method that instantiates a Pipeline object based on the PipelineCreateInputData information passed in.
  • StreamOn(): notifies Pipeline to begin data transfer of hardware
  • StreamOff(): Tells Pipeline to stop data transfer on hardware
  • FinalizePipeline(): Used to complete the Pipeline setup
  • OpenRequest(): Opens a CSL Request for flow
  • ProcessRequest(): Starts sending the Request
  • NotifyNodeMetadataDone(): This method is provided by Pipeline to a Node. When a Node generates metadata, it calls this method to notify the Node that the metadata has completed. Finally, when all nodes notify the Pipeline that metadata has completed, Pipeline will call ProcessMetadataRequestIdDone notify the Session.
  • NotifyNodePartialMetadataDone(): This method is provided by pipelines to nodes. When a Node generates partial metadata, it calls this method to notify the Node that the metadata has completed. Finally, when all nodes notify the Pipeline that metadata has completed, Pipeline will call ProcessPartialMetadataRequestIdDone notify the Session.
  • SinkPortFenceSignaled(): Used to inform the Session that the fence of a sink port is triggered.
  • NonSinkPortFenceSignaled(): Informs the Session that the fence of a non sink port is triggered.

Nodes and connection modes in Pipeline are defined in XML, which mainly contains the following tag definitions:

  • PipelineName: The name used to define the Pipeline
  • NodeList: This tag defines all the nodes of the Pipeline
  • PortLinkages: This tag defines connections between different ports on a Node

5. Node

As a single abstract module with independent processing function, it can be a hardware unit or a software unit. The specific implementation of Node is accomplished by the Node class in CamX, which is mainly divided into two categories in CAMX-Chi. One is the Node implemented by Qualcomm, including hardware Node. One is the Node provided to users for implementation in CHI. The main methods are as follows:

  • Create(): This method is static and instantiates a Node object.
  • ExecuteProcessRequest(): This method is used to send requests to HWL nodes.
  • ProcessRequestIdDone(): This method is called to notify Pipeline once the Node’s current request has been processed.
  • ProcessMetadataDone(): Once the Node’s current request metadata has been generated, this method is called to notify Pipeline.
  • ProcessPartialMetadataDone () : once the Node of the current request of partial metadata have been generated, they call this method to inform to the Pipeline.
  • Create ImageBufferManager CreateImageBufferManager () :

Its customizable parts are defined in XML as tags:

  • NodeName: specifies the name of the Node
  • NodeId: specifies the ID of the Node. The IPE NodeId is 65538, the IFE NodeId is 65536, and the user-defined NodeId is 255.
  • NodeInstance: The name used to define the current instance of the Node.
  • NodeInstanceId: Specifies the Id of the Node instance.

6. Link

It is used to define connections between different ports. A Port can establish multiple connections with other ports belonging to different nodes as required. It is defined by labels, including input ports and output ports. A Link contains an SrcPort and a DstPort, representing the input port and the output port respectively, and BufferProperties is used to represent the buffer configuration between the two ports.

7. Port

As the input/output ports of Node, tags are used to define an InputPort and tags are used to define an OutputPort in XML files. Each Node can use one or more input/output ports as required. Outputports and InputPort structures are used to define these ports in code. Port

  • PortId: indicates the Id of the port: indicates the name of the port
  • NodeName: indicates the name of the Node to which the port belongs
  • NodeId: INDICATES the Id of the Node to which the port belongs
  • NodeInstance: indicates the instance name of the Node to which the port belongs
  • NodeInstanceId: indicates the Id of the Node instance to which the port belongs

Three, component structure relationship

Through the introduction, we for a few basic components have a clearer understanding, but any framework is not only on the component piled up at random, on the contrary, they must be based on their location, according to their unique behavior model, at the same time, in accordance with the contract, commonly known as a series of rules together, together to complete the whole framework of a particular function. So you have to wonder, how are they organized in this framework? What is the relationship between them? Let’s start with the following figure for analysis:

As can be seen from the figure above, they are combined through inclusion relation. Usecase contains Feature, which contains Session, which in turn maintains the flow of internal Pipeline, and each Pipeline connects all nodes through Link. Let’s take a closer look at these relationships:

First, a Usecase represents a particular image acquisition scenarios, such as portrait, rear photographed scene, and so on, at the time of initialization through some specific information according to the top of the incoming to create, the process, on the one hand, instantiate the specific Usecase, this instance is used to manage all the resources the whole scene, It is also responsible for the business processing logic, on the other hand, it gets the specific Usecase defined in XML, and it gets the pipeline used to implement certain functions.

Secondly, Feature is optional in Usecase. If the current user chooses HDR mode or needs to take photos under Zoom or other special functions, one or more features will be created as required during the creation of Usecase. Generally, a Feature corresponds to a specific function. If a scene does not need any specific function, it is completely unnecessary to use or create any Feature.

Then, each Usecase or Feature can contain one or more sessions, and each Session directly manages and is responsible for the data flow of internal Pipeline. Each Request is processed by Usecase or Featuret sent to the internal Pipeline through Session. After data processing is completed, the result is sent to CHI through Session method. It is up to CHI to decide whether to directly send the data to the upper layer or to encapsulate the data and send it to another Session for post-processing.

The relationship between sessions and pipelines is one-to-many. Usually a Session contains only one Pipeline for a specific image processing function, but not always. For example, the Session in FeatureMFNR contains three pipelines. Another example is the post-portrait preview, which also uses a Session to contain two pipelines respectively for the master and deputy double-shot preview, mainly depending on the number of pipelines required by the current function and whether there is a certain correlation between them.

At the same time, according to the above definition of Pipeline, it contains a certain number of nodes, and the more complex the functions implemented, the more nodes it contains, and the more complex the connections between nodes are. For example, the realization of the virtual effect of the rear portrait preview is to get the primary and secondary images taken by RTBOfflinePreview this Pipeline to synthesize two frames of images into a frame with the virtual effect of the image, so as to complete the virtual function.

Finally, the connection mode of nodes in Pipeline is described by Link in XML file. Each Link defines an input and output port corresponding to the input and output ports of different nodes. In this way, the output terminal of one Node and the input terminal of another Node are combined. One by one, when the image data is input from the beginning of the Pipeline, the flow can be carried out from Node to Node according to the defined trajectory, and the data will be processed internally after each Node in the flow process. In this way, when data flows from the beginning to the output of the last Node, the data has been processed many times, and these processing effects are finally added together to achieve the functions of the Pipeline, such as noise reduction, blur, etc.

Iv. Detailed explanation of key processes

1. The Camera Provider starts initialization

When the system starts, the Camera Provider main program will be run. During the initialization process, the get_number_of_camera interface will be called from the obtained camerA_module_t to obtain the number of cameras supported by the underlying program. So there are a lot of initialization actions in CAMX-Chi, as shown below:

The main process is as follows:

  1. Through HAL3Module: : GetInstance () method to instantiate the static HAL3Module object, in the construction method by HwEnvironment: : GetInstance () static method and instantiate the HwEnvironment object, in the constructor, Instantiate the SettingsManager object, Then obtained by OverrideSettingsFile object in its construction method in the/vendor/etc/camera/camoverridesettings. The platform specific configuration information in the TXT file (through this Override mechanism convenient platform vendor to join the custom Configuration), where platform-specific configuration items can be added, such as setting the multiCameraEnable value to indicate whether the current platform supports multi-capture, or setting the overrideLogLevels to configure the Log output level for the CamX-Chi section, and so on.
  2. At the same time, the HwEnvironment constructor will call its Initialize method, which instantiates the CSLModeManager object, and through the interface provided by the CSLModeManager, obtain all the underlying hardware device information. It includes Camera Request Manager, CAPS module (the driver module is mainly used for CSL to obtain the driver information of Camera platform and power control of IPE/BPS module), Sensor/IPE/Flash and other hardware modules. And by calling CSLHwInternalProbeSensorHW method to get the current installation of Sensor module, information, and to save the information of short duration, wait for later stages, Generally speaking, in the process of HwEnvironment initialization, all the underlying hardware driver modules are obtained through the detection method, and their information is stored for subsequent use.
  3. By calling after HwEnvironment ProbeChiCompoents method of object in/vendor/lib64 / camera/components directory search for each Node generated So libraries, and obtain the standard external interface provided by the Node, These nodes not only include some user-defined modules for CHI, but also some hardware modules for CamX implementation, which are finally stored in the ExternalComponentInfo object to be used in the subsequent stages.

In addition, another important operation in the initialization stage is that CamX and CHI obtain each other’s entry method through dlopen each other’s So library, and finally obtain each other’s operation method set through each other’s entry method, and then communicate with each other through these operation methods. The main process is shown in the following figure:

As you can see from the above figure, the HAL3Module constructor loads the com.qti.chi.override.so library through the dlopen method and maps the chi section’s chi_hal_override_entry through dlSYm. And call this method to pass the HAL3Module object member variable m_ChiAppCallbacks(CHIAppCallbacks) into CHI, which contains many function Pointers corresponding to the methods in the operation method set of the CHI part. Once entered into CHI, The function addresses in the CHI local action method set are assigned to m_ChiAppCallbacks so that CamX can later call the CHI method through this member variable, thus maintaining communication with CHI.

Similarly, when ExtensionModule in CHI is initialized, its constructor will also load camera.qcom.so library by calling dlopen method, and map its entry method ChiEntry through DLSYM, and then call this method. G_chiContextOps (ChiContextOps, a structure in which many pointer functions are defined) is passed as a parameter to CamX. Once inside CamX, the local method address is assigned to each function pointer in g_chiContextOps in turn. CHI can then access the CamX method via g_chiContextOps.

2. Start or initialize the camera device

Once the user has to start the Camera application, the App will go to call the CameraManager openCamera method, this method will call to the Camera after the Service of the CameraService: : connectDevice method, The HIDL interface ICameraDevice:: Open () notifies the Provider, and inside the Provider obtains a Camera device by calling the open method of methods in camerA_module_t. Corresponding to the camera3_device_t structure in HAL, the Provider then calls the initialize method of the obtained Camera3_device_t for initialization. Camx-chi’s open and Initialize implementations are as follows:

a) open

This method is the standard method of CamerA_module_T, which is mainly used to obtain camera3_device_t device structure. Camx-chi implements it. The main work of open method is as follows:

  1. Pass the current camera ID into CHI for remap operation. Of course, this remap operation logic is completely based on the needs of users in CHI. Users can add custom Remap logic into CHI according to their own needs.
  2. Instantiate the HALDevice object and call the Initialize method in its constructor, which populates the custom Camera3Device structure in CamX.
  3. M_HALCallbacks. Process_capture_result points to the local method ProcessCaptureResult and m_HALCallbacks. Notify_result points to the local method Notify Register m_HALCallbacks to CHI and pass data or events back to CamX through these two callback methods once CHI data processing is complete.
  4. Finally, the Camera3Device member variable in HALDevice is returned to the Provider in CameraCaptureSession.

HwDevice corresponds to hw_DEVICe_t in Camera3_DEVICe_t, and Camera3DeviceOps corresponds to camera3_device_ops_t. During HALDevice initialization, g_camera3DeviceOps, the structure of the HAL3 interface implemented by CamX, is assigned to Camera3DeviceOps.

b) initialize

This method is called immediately after the open call and is mainly used to pass the upper-level callback interface into HAL. Once data or events are generated, CamX will upload the data or events to the caller through these callback interfaces. The internal implementation is relatively simple.

The initialize method has two parameters: the camera3_device_t structure obtained from the open method and the CameraDevice that implements the camera3_callback_ops_t. Obviously the camera3_device_t structure is not the focus, so the main job of this method is to associate camera3_Callback_OPs_t with CamX, Once the data is ready, the callback method in camera3_callback_ops_t is used to send the data back to CameraDevice in Camera Provider. The basic process can be summarized as follows:

  1. Instantiate a Camera3CbOpsRedirect object and add it to the g_hal3entry.m_cbopslist queue so that it can be retrieved later when needed.
  2. Local process_capture_result and notify methods address assigned to Camera3CbOpsRedirect. CbOps process_capture_result and notify in a function pointer.
  3. The upper into the callback method structure pointer pCamera3CbOpsAPI assigned to Camera3CbOpsRedirect. PCbOpsAPI, and Camera3CbOpsRedirect. Give pCamera3CbOpsAPI cbOps assignment, Pass pCamera3CbOpsAPI to the m_pCamera3CbOps member variable in HALDevice via the Initialize method of JumpTableHal3, Thus m_pCamera3CbOps in HALDevice points to the native methods process_capture_result and notify in CamX.

After doing this, as soon as CHI has data coming in, it first goes to the local method ProcessCaptureResult, where it gets the HALDevice member variable m_pCamera3CbOps, Then call the process_capture_result method in m_pCamera3CbOps, the process_Capture_result method defined in camxhal3entry.cpp, . Then this method will call JumpTableHAL3 process_capture_result method, this method will eventually go to call Camera3CbOpsRedirect. PCbOpsAPI process_capture_result method, This calls back to the callback method passed in from the Provider, and the data is smoothly fed into CameraCaptureSession.

3. Configure the data flow for the camera

In the process of open the Camera application, App after capturing and open the Camera equipment, invokes the CameraDevice. CreateCaptureSession to obtain CameraDeviceSession, and through the Camera API v2 standard interface, Inform the Camera Service, call its CameraDeviceClient. EndConfigure method, Within the method again go through HIDL interface ICameraDeviceSession: : configureStreams_3_4 notify the Provider start dealing with the configuration requirements, within the Provider, The configure_streams method of the Camera3_DEVICe_t structure obtained in the open process is passed into CamX-CHI to configure the data stream. The HAL3 interface configure_STREAMS is implemented by CamX-Chi.

Configuration data flow is an important part of the camX-CHI process, which consists of two phases:

  1. Choose UsecaseId
  2. Create a Usecase based on the selected UsecaseId

Next, we will introduce these two stages in detail:

1) choose UsecaseId

Different UsecaseId respectively corresponding to the different application scenarios, this phase is by calling UsecaseSelector: : GetMatchingUsecase () method to implement, UsecaseId is selected from the operation_mode, num_streams configuration data streams passed in, and the number of sensors currently in use. For example, when numPhysicalCameras is greater than 1 and num_streams is greater than 1, UsecaseId::MultiCamera is selected, indicating that the current dual-camera scenario is used.

Class Usecase is the base Class of all USecases, which defines and implements some common interfaces. CameraUsecaseBase inherits from Usecase and extends some of its functionality. AdvancedCameraUsecase inherits from CameraUsecaseBase as the Usecase implementation class responsible for most of the scenes. UsecaseMultiCamera, inherited from AdvancedCameraUsecase, is now provided to take care of the implementation.

In addition to the double scenario, the other is used for most scenarios AdvancedCameraUsecase class to manage all the resources, we focus on combing the AdvancedCameraUsecase: : Create () method. In AdvancedCameraUsecase: : to do a lot of initialized in the Create method, which includes the following several stages:

  1. Gets the Usecase configuration information from the XML file
  2. Create a Feature
  3. Save the data stream and rebuild the configuration information of the Usecase
  4. Call the Initialize method of the parent class CameraUsecaseBase to do some general initialization

Next, we analyze the stages one by one: 1. The Usecase in the XML file configuration information This part mainly by calling CameraUsecaseBase: : GetXMLUsecaseByName method is implemented. The main operation of this method is to find the Usecase that matches the given usecaseName from the PerNumTargetUsecases array and return it as the return value to the caller. Here we take “UsecaseZSL “as an example to analyze. PerNumTargetUsecases is defined in g_pipeline. H, This file generates g_pipeline.h by converting the contents of common_usecase.xml defined in the platform directory during compilation using the usecaseconverter.pl script.

If Feature is selected for the current scenario, call FeatureSetup to complete the creation. This method mainly determines which features need to be selected through information such as OPERATION_mode, number of cameras and UsecaseId. The specific logic is relatively clear. Once the Feature needs to be used is determined, The corresponding Feature Create() method is called to initialize the Feature.

3. Save the data stream and reconstruct the configuration information of Usecase. The data stream imported from Camera Service needs to be stored for subsequent use. These two steps are implemented mainly through the SelectUsecaseConfig method.

Which is mainly to call the following two methods to achieve:

  • ConfigureStream: This method stores data stream Pointers from the upper configuration into AdvancedCameraUsecase, including m_pPreviewStream for preview and m_pSnapshotStream for taking photos.
  • BuildUsecase: Create a new Usecase and store it in the m_pChiUsecase member variable in AdvancedCameraUsecase. Followed by SetPipelineToSessionMapping method will pipeline associated with the Session.

4. Call the Initialize method of the parent class CameraUsecaseBase to do some general initialization

There are three main operations in this method:

  • Set the Session callback
  • Create a Pipeline
  • Create a Session

Set the Session callback

This method takes two arguments, the second is the default and the first is ChiCallBacks, which is used as a callback for each Session that is created. When all the pipelines in the Session run out, this method is called back to deliver data to the CHI.

Start creating each Pipeline from the previously obtained Pipeline information by calling the CreatePipeline() method.

CreateSession() is used to register the AdvancedCameraUsecase callback function into the Session. Once the Session is processed, A callback is invoked to pass the data back to AdvancedCameraUsecase.

The configure_stream process can be summarized as follows:

  1. UsecaseId is selected according to the operation_mode, camera number, and stream configuration information
  2. Based on the UsecaseId selected, an AdvancedCameraUsecase object is created using the UsecaseFactory simple factory class to manage all the resources in the entire scenario.
  3. The AdvancedCameraUsecase object is created by calling its Create() method, which takes the Usecase configuration information defined by common_usecase.xml, After that, the Feature is created and the pipeline required by the Feature is selected as required, and the pipeline required in the Feature is added into the reconstructed Usecase through Override mechanism.
  4. CameraUsecaseBaese initialize creates each pipeline and Session, and registers the AdvancedCameraUsecase member into the Session. Used for sessions to return data to the Usecase

4. Handle photo requests

When the user opens the camera app to preview or clicks take a photo, it triggers a photo request, CameraDeviceSession Capture or setRepeatingRequest to send the request to the Camera Service through Camera API V2 interface. Then the Camera within the Service: send the request to the CameraDevice: RequestThread thread processing, once into the thread, Will eventually through HIDL interface ICameraCaptureSession: processCaptureRequest_3_4 sends a request to the Provider, after when the Provider after the receipt of a request, Process_capture_request, which calls the camera3_device_t structure, starts HAL’s processing of the Request, which is implemented by CamX-Chi. Now let’s see how camX-Chi implements this method:

First, CamX will forward this request to HALDevice. Chi_override_process_request method (chi_override_process_request method definition bit) is then called through the HALDevice object of the CHI part callback interface m_ChiAppCallbacks. Chi_override_process_request method In chxextensioninterface. CPP) sends the request to CHI.

The chi_override_process_REQUEST method gets the ExtensionModule object and sends the request to the ExtensionModule object, which stores the Usecase object created earlier, and then calls it layer by layer, The ExecuteCaptureRequest method of AdvancedCameraUsecase, which is responsible for processing the Request, is called as follows:

There are two main branches in the AdvancedCameraUsecase ExecuteCaptureRequest that handle each:

  • If there is no current Feature to implement, will leave the default process, according to the above, as shown in the flow chart here will call CameraUsecaseBase: : ExecuteCaptureRequest method, in this method, the first request will be taken out, Repackage the CHICAPTUREREQUEST, then call the CheckAndActivatePipeline method to wake up the pipeline. This operation is finally called to the Session’s StreamOn method, after waking up the pipeline, continue execution, Then encapsulated in a Request sent to the CamX after final call to the corresponding Session: : ProcessCaptureRequest method, the Request into a Session for internal circulation.
  • If the current scenario needs to implement a Feature, the Feature’s ExecuteProcessRequest method is called directly to send the request to the Feature for processing. Finally still will call into the Session: : StreamOn and Session: : ProcessCaptureRequest method to complete awakening pipeline and issued the request to the operation of the Session.

The process can eventually calls to two more critical method Session: : StreamOn and Session: : ProcessCaptureRequest, then under these two methods are introduced:

Session::StreamOn

From the name of the method, we can know that this method is mainly used to start hardware data output. Specifically, it is to configure the Sensor register to start drawing, and inform each Node of the current Session status, so that they are also ready to process data internally. Therefore, the subsequent flow of relevant Request is based on the premise of this method, so the importance of this method can be seen, and its operation process is shown in the following figure:

Session’s StreamOn method does two things:

  • Call the FinalizeDeferPipeline() method, or the FinalizePipeline method of the pipeline if the current pipeline is not initialized, This method performs FinalizeInitialization, CreateBufferManager, NotifyPipelineCreated, and PrepareNodeStreamOn operations for each Node that is subordinate to the current pipeline FinalizeInitialization is used to complete Node initialization, and NotifyPipelineCreated is used to notify Node of the current Pipeline status. At this time, Node can perform corresponding operations according to its own needs. PrepareNodeStreamOn method is mainly to complete the configuration of control hardware module of Sensor and IFE Node before drawing, including the setting of exposure parameters. The CreateBufferManagers method refers to a very important Buffer management mechanism in CamX-Chi for the creation of Node’s ImageBufferManager, This class is used to manage buffer requests/flows/releases of output ports in nodes.
  • Call Pipeline’s StreamOn method, which further tells CSL to start streaming, and call each Node’s OnNodeStreamOn method, This method calls ImageBufferManager Activate(), which actually allocates the buffer used to load the image data, and then calls CHI’s user-defined Nod pOnStreamOn method, The user can do some custom operations in this method.

Session: : ProcessCaptureRequest for every Request flow, only at the entrance of the method, the specific process as below:

The above process can be summarized as the following steps:

  1. Session is entered by calling the ProcessCaptureRequest method of the Session, and the ProcessRequest method of the Pipeline is called to tell the Pipeline to start processing the Request.
  2. In Pipeline, the SetupRequest method of each internal Node will be called first to set the Output Port and Input Port of the Node respectively. Then add all nodes to the DRQ by calling the AddDeferredNode method. There are two queues in DRQ, m_readyNodes for storing nodes with no dependencies and m_deferredNodes for storing nodes waiting for the dependency relationship to be satisfied. When the DispatchReadyNodes method of DRQ is called, ProcessRequest will start processing the request from the m_readyNodes queue. Meta data will be updated to DRQ during processing. When the Node is finished processing, The undependent nodes in m_deferredNodes are moved to m_readyNodes and the DispatchReadyNodes method is called again to fetch the Node from m_readyNodes for processing.
  3. In this process, after the data processing of Node is completed, it will be notified to Pipeline through CSLFenceCallback. At this time, Pipeline will judge whether the Output port of the current Node is a Sink port (Output to CHI). If not, Then, the dependency will be updated to DRQ, and the Node without the dependency will be moved to m_readyNodes queue, and then call DispatchReadyNdoes to continue the flow into DRQ. If it is Sink Port, it means that this Node is the end of the whole Pipeline. Call sinkPortFenceSignaled to send data to the Session, and finally send the result to the CHI by calling NotifyResult in the Session.

DeferredRequestQueue

DeferredRequestQueue is a new queue in DeferredRequestQueue. DeferredRequestQueue inherits from IPropertyPoolObserver, Implements the OnPropertyUpdate/OnMetadataUpdate/OnPropertyFailure/OnMetadataFailure interface, this a few interface used to receive updates, Meta Data and the Property on the other hand, DRQ includes the following main methods:

  • Create() this method is used to Create a DRQ that creates a m_pDependencyMap for storing dependency information and registers itself in MetadataPool. Updates to meta data or property are notified to the DRQ through several interfaces implemented in the class.

  • DispatchReadyNodes() Takes nodes from the m_readyNodes queue and sends them to the m_hDeferredWorker thread for processing.

  • AddDeferredNode() is used to add a dependency to m_pDependencyMap.

  • FenceSignaledCallback() notifies DRQ via a series of callbacks after Node completes processing of a request. In this method, UpdateDependency is called first to update the dependency. A call to DispatchReadyNodes triggers the start of processing for nodes in the ready state

  • OnPropertyUpdate() This method is defined in the IPropertyPoolObserver interface and implemented by DRQ to receive notifications of Property updates and internally call UpdateDependency to update the dependency.

  • OnMetadataUpdate() This method is defined in the IPropertyPoolObserver interface and implemented by DRQ to receive notifications of Metadata updates and internally call UpdateDependency to update dependencies.

  • UpdateDependency() This method is used to update Node dependency information and move non-dependent nodes from m_deferredNodes queue to m_readyNodes, This allows the Node to run after a subsequent call to DispatchReadyNodes.

  • DeferredWorkerWrapper() is the handler of the m_hDeferredWorker thread. It is used to process nodes that need to issue a request, update the dependency again, and then call DispatchReadyNodes again to start processing.

Note that Pipeline is first added to the DRQ for each Node by calling AddDeferredNode. All nodes are added to m_readyNodes and then dispatchReadyNodes. The DRQ is triggered to start the entire internal processing process. The basic process can be seen in the following figure, which is used for further combing:

  1. When the DRQ dispatchReadyNodes method is called, the Dependency is removed from the m_readyNodes linked list and posted to the DeferredWorkerWrapper thread. After the thread takes the Node from the Dependency and calls its ProcessRequest method to start processing the request inside the Node, if the Dependency still exists at the end of processing, Then call the AddDeferredNode method to add Node to the m_deferredNodes linked list and add the new dependency to the M_PDeferredMap Hash table.
  2. Meta Data and properties are continuously updated as Node processes the request, which is updated to MetadataPool by calling the MetadataSlot PublishMetadata method. MetadataPool notifies DRQ of new meta data and property updates by calling the OnPropertyUpdate and OnMetadataUpdate callback methods that were registered during DRQ initialization. The UpdateDependency method is then called to update meta Data and property to m_pDependencyMap. Add the Node without any dependencies from m_deferredNodes to m_readyNodes and wait for processing.
  3. At the same time, The result of Node’s processing is notified to pipeline via ProcessFenceCallback and pipeline’s NonSinkPortFenceSignaled method is called, This method calls DRQ’s FenceSignaledCallback method, which in turn calls UpdateDependency to update dependencies and adds nodes with all dependencies from m_deferredNodes to m_readyNodes. Then call dispatchReadyNodes to continue processing.

5. Upload photo results

After the user starts the camera application, the camera frame will start to process it after receiving a Request. Once the image data is generated, it will be returned to the application layer for display through layer upon layer callback. Here, we conduct a simple sorting for the uploading process of the photo results in camX-Chi:

Each Request corresponds to three results, partial metadata, metadata and image data. For each Result, the upload process can be roughly divided into the following two stages:

  • The Session processes the image data internally and sends the results to the Usecase
  • Usecase receives data from the Session and uploads it to the Provider

Let’s take a look at how the Session sends the results to Usecase after the image data is processed internally:

In the process of the whole requets circulation, once with Partial Meta Data generated in the Node, will call Node ProcessPartialMetadataDone method to notify affiliate of Pipeline, Its internal call again NotifyNodePartialMetadataDone method of pipeline. Each call to the Pipeline NotifyNodePartialMetadataDone methods to add pPerRequestInfo – numNodesPartialMetadataDone a and judge whether the current value is equal to the number of Node in Pipeline Once, is equal, illustrate the current Node all completed a partial meta data update action, right now, will call ProcessPartialMetadataRequestIdDone method, it will be to remove partial meta data, In addition, it is reencapsulated into a ResultsData structure and passed into the Session as a parameter through the NotifyResult method of the Session. Later in the Session through layer upon layer calls eventually call to internal member variable m_chiCallBacks ChiProcessPartialCaptureResult method, this method is the time to create the Session, Introduced to the Session of the Usecase method (AdvancedCameraUsecase: : ProcessDriverPartialCaptureResultCb), the method will return to the CHI meta data.

Similarly, the logic of Metadata is similar to Partial Metadata. Each Node calls the ProcessMetadataDone method to send data to the Pipeline during the processing of a request. Once all meta data of nodes have been sent, the pipeline calls NotifyNodeMetadataDone to send the final result to the Session. The ChiProcessCaptureResult method of the Session member variable m_chiCallBacks is called, sending the result to Usecase in CHI.

The flow of image data is a little different from the flow of the previous two meta data. Once the processing of image data inside Node is completed, its ProcessFenceCallback method will be called, in which it will check whether the current output is the SInk Buffer. If so, the SinkPortFenceSignaled method of Pipeline will be called to send data to the Pipeline. In this method, the Pipeline will send data to the Session. Finally, after layer upon layer invocation, The ChiProcessCaptureResult method of the Session member variable m_chiCallBacks is called, sending the result to Usecase in CHI.

Let’s look at how Usecase sends Session data to the Provider once it receives it:

We take the commonly used AdvancedCameraUsecase as an example to comb the code:

As shown in the figure above, the flow logic of the whole result is relatively clear. CamX sends the result back to CHI through callback method, and in CHI, it first determines whether it needs to be sent to the specific Feature. If so, Call the corresponding Feature ProcessDriverPartialCaptureResult or ProcessResult method result will be sent to the specific Feature, once processing is complete, Will call call CameraUsecaseBase ProcessAndReturnPartialMetadataFinishedResults and send the result to the Usecase ProcessAndReturnFinishedResults method, If it does not need to be sent to Feature for processing, In the AdvancedCameraUsecase call CameraUsecaseBase SessionCbPartialCaptureResult and SessionCbCaptureResult method, Then through the Usecase: : send the result to the ExtensionModule ReturnFrameResult method, Process_capture_result, the callback function in CamX stored in ExtensionModule, is then called to send the result to HALDevice in CamX, which in turn passes in the callback method passed in from the previous store. Finally send the results to CameraDeviceSession.

Through the above sorting, it can be found that the whole CAMX-Chi framework is well designed, with a clear directory structure, simple and efficient framework and clear flow control logic. For example, for a certain image request, The whole process goes through Usecase, Feature, Session, Pipeline and is processed in specific Node, and the final output results. In addition, compared with the previous QCamera & MM-Camera framework, the extension of a certain algorithm requires the custom modification to be embedded in the whole process code, camX-Chi improves its scalability and lowers the development threshold by putting the custom implementation into CHI. Platform vendors who are not familiar with the CamX framework can successfully add new features through minor modifications. But no one is perfect, and frameworks, too, are too asynchronous, making it harder to locate and solve problems, and putting a lot of pressure on developers. In addition, the frame has a high requirement for memory, so in some low-end models, especially those with low memory, the efficiency of the whole frame may be limited to some extent, resulting in lower than expected camera efficiency

The original link: blog.csdn.net/u012596975/…

Friendly recommendation: Android dry goods to share

At this point, this has ended, if there is wrong place, welcome your suggestion and correction. Meanwhile, I look forward to your attention. Thank you for reading. Thank you!