Mono Image Sensor Processing pipeline - 2024.1 English

Vitis Libraries

Release Date
2024-08-06
Version
2024.1 English

The Mono image sensor is different to the RGB Bayer sensor. Some applications do not need color information. In such cases, you can use the mono image sensor instead of the color sensor. The mono image sensor pipeline has a lot of advantages over the color sensor processing: computational cost and higher resolution because of a single channel, and also reduced errors while doing image reconstruction using demosaic in the color sensor processing.

This ISP includes following blocks:

  • Black level correction: Black level leads to the whitening of image in dark region and perceived loss of overall contrast. The Blacklevelcorrection algorithm corrects the black and white levels of the overall image.
  • BPC (Bad pixel correction): Using median filter for BPC. An image sensor may have a certain number of defective/bad pixels that may be the result of manufacturing faults or variations in pixel voltage levels based on temperature or exposure. Bad pixel correction module removes defective pixels.
  • Gain Control: The Gain control module improves the overall brightness of the image.
  • Quantization and Dithering: Quantization and Dithering performs the uniform quantization to also reduce higher bit depth to lower bit depths.
  • Gamma correction: Gamma correction improves the overall brightness of image.
  • Autoexposure correction: Using the CLAHE algorithm to improve brightness and contrast of the image.

isp-mono

Current design example demonstrates how to use ISP functions in a pipeline.

You can dynamically configure the following parameters to the pipeline.

Table 243 Runtime Parameters for the Pipeline
Parameter Description
lgain To configure gain value for the luminence channel.
gamma_lut Lookup table for gamma values.
rows The number of rows in the image or height of the image.
cols The number of columns in the image or width of the image.
clip clip is used to set the threshold for contrast limit in the processing
tilesY The image is divided into tiles in the CLAHE. The tilesY represents the number of tiles in Y direction.
tilesX The image is divided into tiles in the CLAHE. The tilesY represents the number of tiles in X direction.

You can also use the following compile-time parameters to the pipeline.

Table 244 Compile-Time Parameters for the Pipeline
Parameter Description
XF_HEIGHT Maximum height of input and output image
XF_WIDTH Maximum width of input and output image (Must be a multiple of NPC)
XF_SRC_T Input pixel type,Supported pixel widths are 8,10,12,16

The following example demonstrates the ISP pipeline with the above list of functions.

         void ISPPipeline_accel(ap_uint<INPUT_PTR_WIDTH>* img_inp,
                                            ap_uint<OUTPUT_PTR_WIDTH>* img_out,
                                            int height,
                                            int width,
                                            uint16_t lgain,
                                            unsigned char gamma_lut[256],
                                            int clip,
                                            int tilesY,
                                            int tilesX) {

         #pragma HLS INTERFACE m_axi     port=img_inp  offset=slave bundle=gmem1
         #pragma HLS INTERFACE m_axi     port=img_out  offset=slave bundle=gmem2
         #pragma HLS INTERFACE m_axi      port=gamma_lut offset=slave  bundle=gmem3 depth=256

         #pragma HLS INTERFACE s_axilite  port=clip
         #pragma HLS INTERFACE s_axilite  port=tilesY
         #pragma HLS INTERFACE s_axilite  port=tilesX
         #pragma HLS INTERFACE s_axilite  port=return

         #pragma HLS ARRAY_PARTITION variable=_lut1 dim=3 complete
         #pragma HLS ARRAY_PARTITION variable=_lut2 dim=3 complete


                         if (!flag) {
                                 ISPpipeline(img_inp, img_out, height, width, lgain, gamma_lut, _lut1, _lut2, _clipCounter, clip, tilesX,
                                                         tilesY);
                                 flag = 1;

                         } else {
                                 ISPpipeline(img_inp, img_out, height, width, lgain, gamma_lut, _lut2, _lut1, _clipCounter, clip, tilesX,
                                                         tilesY);
                                 flag = 0;
                         }
                 }

         void ISPpipeline(ap_uint<INPUT_PTR_WIDTH>* img_inp,
  ap_uint<OUTPUT_PTR_WIDTH>* img_out,
  unsigned short height,
  unsigned short width,
  uint16_t lgain,
  unsigned char gamma_lut[256],
  ap_uint<HIST_COUNTER_BITS> _lutw[TILES_Y_MAX][TILES_X_MAX][(XF_NPIXPERCYCLE(XF_NPPC) << 1)]
                                  [1 << XF_DTPIXELDEPTH(XF_LTM_T, XF_NPPC)],
  ap_uint<HIST_COUNTER_BITS> _lutr[TILES_Y_MAX][TILES_X_MAX][(XF_NPIXPERCYCLE(XF_NPPC) << 1)]
                                  [1 << XF_DTPIXELDEPTH(XF_LTM_T, XF_NPPC)],
  ap_uint<CLIP_COUNTER_BITS> _clipCounter[TILES_Y_MAX][TILES_X_MAX],
  int clip,
  int tilesY,
  int tilesX) {

                 #pragma HLS INLINE OFF

                         xf::cv::Mat<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_IN_1> imgInput1(height, width);
                         xf::cv::Mat<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_IN_2> imgInput2(height, width);
                         xf::cv::Mat<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_DPC_OUT> dpc_out(height, width);
                         xf::cv::Mat<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_GAIN_OUT> gain_out(height, width);
                         xf::cv::Mat<XF_DST_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_IMPOP> impop(height, width);
                         xf::cv::Mat<XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_DST> _dst(height, width);
                         xf::cv::Mat<XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_AEC_IN> aecin(height, width);
                         xf::cv::Mat<XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_OUT> _imgOutput(height, width);


                 #pragma HLS DATAFLOW


                         CLAHE_T obj;

                         const int Q_VAL = 1 << (XF_DTPIXELDEPTH(XF_SRC_T, XF_NPPC));

                         float inputMax = (1 << (XF_DTPIXELDEPTH(XF_SRC_T, XF_NPPC))) - 1;

                         float mul_fact = (inputMax / (inputMax - BLACK_LEVEL));
unsigned int blc_config_1 = (int)(mul_fact * 65536); // mul_fact int Q16_16 format
unsigned int blc_config_2 = BLACK_LEVEL;

                         xf::cv::Array2xfMat<INPUT_PTR_WIDTH, XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_IN_1>(img_inp, imgInput1);
                         xf::cv::blackLevelCorrection<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, 16, 15, 1, XF_CV_DEPTH_IN_1, XF_CV_DEPTH_IN_2>(imgInput1, imgInput2, blc_config_2,
                                                                                                                                                                                         blc_config_1);

                         xf::cv::medianBlur<WINDOW_SIZE, XF_BORDER_REPLICATE, XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_IN_2, XF_CV_DEPTH_DPC_OUT>(imgInput2, dpc_out);
                         xf::cv::gaincontrol_mono<XF_SRC_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_DPC_OUT, XF_CV_DEPTH_GAIN_OUT>(dpc_out, gain_out, lgain);

                         if (XF_DST_T == XF_8UC1) {
                                 fifo_copy<XF_DST_T, XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_GAIN_OUT, XF_CV_DEPTH_AEC_IN>(gain_out, aecin, height, width);
                         } else {
                                 xf::cv::xf_QuatizationDithering<XF_DST_T, XF_LTM_T, XF_HEIGHT, XF_WIDTH, 256, Q_VAL, XF_NPPC, XF_CV_DEPTH_GAIN_OUT, XF_CV_DEPTH_AEC_IN>(gain_out, aecin);
                         }

                         obj.process(_dst, aecin, _lutw, _lutr, _clipCounter, height, width, clip, tilesY, tilesX);

                         xf::cv::gammacorrection<XF_LTM_T, XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_DST, XF_CV_DEPTH_OUT>(_dst, _imgOutput, gamma_lut);

                         xf::cv::xfMat2Array<OUTPUT_PTR_WIDTH, XF_LTM_T, XF_HEIGHT, XF_WIDTH, XF_NPPC, XF_CV_DEPTH_OUT>(_imgOutput, img_out);
                 }