Files
QD4C-firmware/libraries/zf_device/zf_device_scc8660_dvp.c
2023-12-11 21:45:06 +08:00

645 lines
29 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*********************************************************************************************************************
* CH32V307VCT6 Opensourec Library 即CH32V307VCT6 开源库)是一个基于官方 SDK 接口的第三方开源库
* Copyright (c) 2022 SEEKFREE 逐飞科技
*
* 本文件是CH32V307VCT6 开源库的一部分
*
* CH32V307VCT6 开源库 是免费软件
* 您可以根据自由软件基金会发布的 GPLGNU General Public License即 GNU通用公共许可证的条款
* 即 GPL 的第3版即 GPL3.0)或(您选择的)任何后来的版本,重新发布和/或修改它
*
* 本开源库的发布是希望它能发挥作用,但并未对其作任何的保证
* 甚至没有隐含的适销性或适合特定用途的保证
* 更多细节请参见 GPL
*
* 您应该在收到本开源库的同时收到一份 GPL 的副本
* 如果没有,请参阅<https://www.gnu.org/licenses/>
*
* 额外注明:
* 本开源库使用 GPL3.0 开源许可证协议 以上许可申明为译文版本
* 许可申明英文版在 libraries/doc 文件夹下的 GPL3_permission_statement.txt 文件中
* 许可证副本在 libraries 文件夹下 即该文件夹下的 LICENSE 文件
* 欢迎各位使用并传播本程序 但修改内容时必须保留逐飞科技的版权声明(即本声明)
*
* 文件名称 zf_device_scc8660_dvp_dvp
* 公司名称 成都逐飞科技有限公司
* 版本信息 查看 libraries/doc 文件夹内 version 文件 版本说明
* 开发环境 MounRiver Studio V1.8.1
* 适用平台 CH32V307VCT6
* 店铺链接 https://seekfree.taobao.com/
*
* 修改记录
* 日期 作者 备注
* 2022-09-15 大W first version
********************************************************************************************************************/
/*********************************************************************************************************************
*接线定义:
* ------------------------------------
* 模块管脚 单片机管脚
* TXD 查看 zf_device_scc8660_dvp.h 中 SCC8660_COF_UART_TX 宏定义
* RXD 查看 zf_device_scc8660_dvp.h 中 SCC8660_COF_UART_RX 宏定义
* D0 查看 zf_device_scc8660_dvp.h 中 SCC8660_D0_PIN 宏定义
* D1 查看 zf_device_scc8660_dvp.h 中 SCC8660_D1_PIN 宏定义
* D2 查看 zf_device_scc8660_dvp.h 中 SCC8660_D2_PIN 宏定义
* D3 查看 zf_device_scc8660_dvp.h 中 SCC8660_D3_PIN 宏定义
* D4 查看 zf_device_scc8660_dvp.h 中 SCC8660_D4_PIN 宏定义
* D5 查看 zf_device_scc8660_dvp.h 中 SCC8660_D5_PIN 宏定义
* D6 查看 zf_device_scc8660_dvp.h 中 SCC8660_D6_PIN 宏定义
* D7 查看 zf_device_scc8660_dvp.h 中 SCC8660_D7_PIN 宏定义
* PCLK 查看 zf_device_scc8660_dvp.h 中 SCC8660_PCLK_PIN 宏定义
* VSYNC 查看 zf_device_scc8660_dvp.h 中 SCC8660_VSY_PIN 宏定义
* HSYNC 查看 zf_device_scc8660_dvp.h 中 SCC8660_HERF_PIN 宏定义
* ------------------------------------
********************************************************************************************************************/
#include "zf_device_type.h"
#include "zf_device_camera.h"
#include "zf_device_config.h"
#include "zf_driver_dvp.h"
#include "zf_driver_delay.h"
#include "zf_driver_soft_iic.h"
#include "zf_device_scc8660_dvp.h"
vuint8 scc8660_finish_flag = 0; // 一场图像采集完成标志位
__attribute__((aligned(4))) uint16 scc8660_image[SCC8660_H][SCC8660_W];
static fifo_struct *camera_receiver_fifo;
static scc8660_type_enum scc8660_type;
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 配置摄像头内部配置信息 内部调用
// 参数说明 buff 发送配置信息的地址
// 返回参数 uint8 1-失败 0-成功
// 使用示例 if(scc8660_set_config(scc8660_set_confing_buffer)){}
// 备注信息 调用该函数前请先初始化串口
//-------------------------------------------------------------------------------------------------------------------
static uint8 scc8660_set_config (const int16 buff[SCC8660_CONFIG_FINISH][2])
{
uint8 uart_buffer[4];
uint8 return_state = 0;
// uint16 temp_value = 0;
uint16 timeout_count = 0;
uint32 loop_count = 0;
uint32 uart_buffer_index = 0;
// 设置参数 具体请参看问题锦集手册
// 开始配置摄像头并重新初始化
for(loop_count = SCC8660_MANUAL_WB; SCC8660_SET_REG_DATA > loop_count; loop_count --)
{
uart_buffer[0] = 0xA5; // 帧头
uart_buffer[1] = buff[loop_count][0]; // 指令
// temp_value = buff[loop_count][1];
uart_buffer[2] = buff[loop_count][1] >> 8; // 数据高低位
uart_buffer[3] = (uint8)buff[loop_count][1]; // 数据高低位
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4); // 数据通过串口发送
system_delay_ms(2);
}
do
{
if(3 <= fifo_used(camera_receiver_fifo)) // 等待 FIFO 中存入三个数据
{
uart_buffer_index = 3; // 长度缓冲变量
fifo_read_buffer(camera_receiver_fifo, // FIFO 指针
uart_buffer, // 读取到指定缓冲
&uart_buffer_index, // 长度缓冲
FIFO_READ_AND_CLEAN); // 读取并清空操作
if((0xff == uart_buffer[1]) || (0xff == uart_buffer[2])) // 判断是否回应的成功
{
return_state = 0;
break;
}
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
// 以上部分对摄像头配置的数据全部都会保存在摄像头上 51 单片机的 eeprom 中
// 利用 set_exposure_time 函数单独配置的曝光数据不存储在 eeprom 中
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 获取摄像头内部配置信息 内部调用
// 参数说明 buff 接收配置信息的地址
// 返回参数 uint8 1-失败 0-成功
// 使用示例 if(scc8660_get_config(scc8660_get_confing_buffer)){}
// 备注信息 调用该函数前请先初始化串口
//-------------------------------------------------------------------------------------------------------------------
static uint8 scc8660_get_config (int16 buff[SCC8660_CONFIG_FINISH-1][2])
{
int8 loop_count = 0;
uint8 uart_buffer[4];
uint8 return_state = 0;
// uint16 temp_value = 0;
uint16 timeout_count = 0;
uint32 uart_buffer_index = 0;
for(loop_count = SCC8660_MANUAL_WB - 1; 0 <= loop_count; loop_count --)
{
uart_buffer[0] = 0xA5; // 帧头
uart_buffer[1] = SCC8660_GET_STATUS; // 获取指令
// temp_value = buff[loop_count][0];
uart_buffer[2] = buff[loop_count][0] >> 8; // 指令高低位
uart_buffer[3] = (uint8)buff[loop_count][0]; // 指令高低位
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4); // 数据通过串口发送
timeout_count = 0;
do
{
if(3 <= fifo_used(camera_receiver_fifo)) // 等待 FIFO 中存入三个数据
{
uart_buffer_index = 3; // 长度缓冲变量
fifo_read_buffer(camera_receiver_fifo, // FIFO 指针
uart_buffer, // 读取到指定缓冲
&uart_buffer_index, // 长度缓冲
FIFO_READ_AND_CLEAN); // 读取并清空操作
buff[loop_count][1] = uart_buffer[1] << 8 | uart_buffer[2]; // 回读数据写入缓冲区
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
if(SCC8660_INIT_TIMEOUT < timeout_count) // 超时
{
return_state = 1;
break;
}
}
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 获取摄像头 ID
// 参数说明 void
// 返回参数 uint16 0-版本号正确 N-版本号错误
// 使用示例 scc8660_uart_check_id(); // 调用该函数前请先初始化串口
// 备注信息
//-------------------------------------------------------------------------------------------------------------------
static uint8 scc8660_uart_check_id (void)
{
uint8 uart_buffer[4];
uint8 return_state = 1;
uint16 timeout_count = 0;
uint32 uart_buffer_index = 0;
if(fifo_used(camera_receiver_fifo))
{
fifo_clear(camera_receiver_fifo);
}
uart_write_byte(SCC8660_COF_UART, 0xFF);
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_GET_WHO_AM_I;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
timeout_count = 0;
do
{
if(3 <= fifo_used(camera_receiver_fifo))
{
uart_buffer_index = 3;
fifo_read_buffer(camera_receiver_fifo, uart_buffer, &uart_buffer_index, FIFO_READ_AND_CLEAN);
return_state = !(0x03 == uart_buffer[2]);
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 SCC8660 通信串口回调函数
// 参数说明 void
// 返回参数 void
// 使用示例 scc8660_uart_handler();
// 备注信息 通过 zf_device_type.c 的接口调用 用户在使用默认设置时不需要关心
//-------------------------------------------------------------------------------------------------------------------
static void scc8660_uart_handler (void)
{
uint8 data = 0;
uart_query_byte(SCC8660_COF_UART, &data);
if(NULL != camera_receiver_fifo)
{
if(0xA5 == data)
{
fifo_clear(camera_receiver_fifo);
}
fifo_write_element(camera_receiver_fifo, data);
}
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 SCC8660 场中断回调函数
// 参数说明 void
// 返回参数 void
// 使用示例 scc8660_vsync_handler();
// 备注信息 通过 zf_device_type.c 的接口调用 用户在使用默认设置时不需要关心
//-------------------------------------------------------------------------------------------------------------------
static void scc8660_dvp_handler (void)
{
scc8660_finish_flag = 1; // 摄像头采集完成标志位置1
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 获取彩色摄像头固件版本 内部调用
// 参数说明 void
// 返回参数 uint16 版本号
// 使用示例 scc8660_get_version(); // 调用该函数前请先初始化摄像头配置串口
// 备注信息
//-------------------------------------------------------------------------------------------------------------------
uint16 scc8660_get_version (void)
{
uint8 uart_buffer[4];
uint16 temp_value = 0;
uint16 timeout_count = 0;
uint16 return_value = 0;
uint32 uart_buffer_index = 0;
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_GET_STATUS;
temp_value = SCC8660_GET_VERSION;
uart_buffer[2] = temp_value >> 8;
uart_buffer[3] = (uint8)temp_value;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
do
{
if(3 <= fifo_used(camera_receiver_fifo))
{
uart_buffer_index = 3;
fifo_read_buffer(camera_receiver_fifo, uart_buffer, &uart_buffer_index, FIFO_READ_AND_CLEAN);
return_value = uart_buffer[1] << 8 | uart_buffer[2];
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
return return_value;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 单独设置图像亮度
// 参数说明 data 需要设置的亮度值
// 返回参数 uint8 1-失败 0-成功
// 使用示例 scc8660_set_brightness(data); // 通过该函数设置的参数不会被51单片机保存
// 备注信息 调用该函数前请先初始化摄像头配置串口
//-------------------------------------------------------------------------------------------------------------------
uint8 scc8660_set_brightness (uint16 data)
{
uint8 uart_buffer[4];
uint8 return_state = 0;
uint16 temp_value = 0;
uint16 timeout_count = 0;
uint32 uart_buffer_index = 0;
if(SCC8660_UART == scc8660_type)
{
do
{
uint8 *receiver_buffer = (uint8 *)malloc(SCC8660_COF_BUFFER_SIZE);
if(NULL == receiver_buffer)
{
zf_log(0, "FIFO buffer malloc error.");
return_state = 1;
break;
}
camera_receiver_fifo = (fifo_struct *)malloc(sizeof(fifo_struct));
if(NULL == camera_receiver_fifo)
{
zf_log(0, "FIFO malloc error.");
free(receiver_buffer);
return_state = 1;
break;
}
fifo_init(camera_receiver_fifo, FIFO_DATA_8BIT, receiver_buffer, SCC8660_COF_BUFFER_SIZE);
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_SET_BRIGHT;
uart_buffer[2] = data >> 8;
uart_buffer[3] = (uint8)data;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
do
{
if(3 <= fifo_used(camera_receiver_fifo))
{
uart_buffer_index = 3;
fifo_read_buffer(camera_receiver_fifo, uart_buffer, &uart_buffer_index, FIFO_READ_AND_CLEAN);
temp_value = uart_buffer[1] << 8 | uart_buffer[2];
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
if((temp_value != data) || (SCC8660_INIT_TIMEOUT <= timeout_count))
{
return_state = 1;
}
if(NULL != receiver_buffer)
{
free(receiver_buffer);
free(camera_receiver_fifo);
camera_receiver_fifo = NULL;
}
}while(0);
}
else
{
return_state = scc8660_sccb_set_brightness(data);
}
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 单独设置白平衡
// 参数说明 data 需要设置的亮度值
// 返回参数 uint8 1-失败 0-成功
// 使用示例 scc8660_set_white_balance(data); // 调用该函数前请先初始化摄像头配置串口
// 备注信息 通过该函数设置的参数不会被51单片机保存
//-------------------------------------------------------------------------------------------------------------------
uint8 scc8660_set_white_balance (uint16 data)
{
uint8 uart_buffer[4];
uint8 return_state = 0;
uint16 temp_value = 0;
uint16 timeout_count = 0;
uint32 uart_buffer_index = 0;
if(SCC8660_UART == scc8660_type)
{
do
{
uint8 *receiver_buffer = (uint8 *)malloc(SCC8660_COF_BUFFER_SIZE);
if(NULL == receiver_buffer)
{
zf_log(0, "FIFO buffer malloc error.");
return_state = 1;
break;
}
camera_receiver_fifo = (fifo_struct *)malloc(sizeof(fifo_struct));
if(NULL == camera_receiver_fifo)
{
zf_log(0, "FIFO malloc error.");
free(receiver_buffer);
return_state = 1;
break;
}
fifo_init(camera_receiver_fifo, FIFO_DATA_8BIT, receiver_buffer, SCC8660_COF_BUFFER_SIZE);
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_SET_MANUAL_WB;
uart_buffer[2] = data >> 8;
uart_buffer[3] = (uint8)data;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
do
{
if(3 <= fifo_used(camera_receiver_fifo))
{
uart_buffer_index = 3;
fifo_read_buffer(camera_receiver_fifo, uart_buffer, &uart_buffer_index, FIFO_READ_AND_CLEAN);
temp_value = uart_buffer[1] << 8 | uart_buffer[2];
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
if((temp_value != data) || (SCC8660_INIT_TIMEOUT <= timeout_count))
{
return_state = 1;
}
if(NULL != receiver_buffer)
{
free(receiver_buffer);
free(camera_receiver_fifo);
camera_receiver_fifo = NULL;
}
}while(0);
}
else
{
return_state = scc8660_sccb_set_manual_wb(data);
}
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 对摄像头内部寄存器进行写操作
// 参数说明 addr 摄像头内部寄存器地址
// 参数说明 data 需要写入的数据
// 返回参数 uint8 0-成功 1-失败 2-内存申请失败
// 使用示例 scc8660_set_reg(addr, data); // 调用该函数前请先初始化串口
// 备注信息
//-------------------------------------------------------------------------------------------------------------------
uint8 scc8660_set_reg (uint8 addr, uint16 data)
{
uint8 uart_buffer[4];
uint8 return_state = 0;
uint16 temp_value = 0;
uint16 timeout_count = 0;
uint32 uart_buffer_index = 0;
if(SCC8660_UART == scc8660_type)
{
do
{
uint8 *receiver_buffer = (uint8 *)malloc(SCC8660_COF_BUFFER_SIZE);
if(NULL == receiver_buffer)
{
return_state = 2;
break;
}
camera_receiver_fifo = (fifo_struct *)malloc(sizeof(fifo_struct));
if(NULL == camera_receiver_fifo)
{
free(receiver_buffer);
return_state = 2;
break;
}
fifo_init(camera_receiver_fifo, FIFO_DATA_8BIT, receiver_buffer, SCC8660_COF_BUFFER_SIZE);
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_SET_REG_ADDR;
uart_buffer[2] = 0x00;
uart_buffer[3] = (uint8)addr;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
system_delay_ms(10);
uart_buffer[0] = 0xA5;
uart_buffer[1] = SCC8660_SET_REG_DATA;
temp_value = data;
uart_buffer[2] = temp_value >> 8;
uart_buffer[3] = (uint8)temp_value;
uart_write_buffer(SCC8660_COF_UART, uart_buffer, 4);
do
{
if(3 <= fifo_used(camera_receiver_fifo))
{
uart_buffer_index = 3;
fifo_read_buffer(camera_receiver_fifo, uart_buffer, &uart_buffer_index, FIFO_READ_AND_CLEAN);
temp_value = uart_buffer[1] << 8 | uart_buffer[2];
break;
}
system_delay_ms(1);
}while(SCC8660_INIT_TIMEOUT > timeout_count ++);
if((temp_value != data) || (SCC8660_INIT_TIMEOUT <= timeout_count))
{
return_state = 1;
}
if(NULL != receiver_buffer)
{
free(receiver_buffer);
free(camera_receiver_fifo);
camera_receiver_fifo = NULL;
}
}while(0);
}
else
{
return_state = scc8660_sccb_set_reg(addr, data);
}
return return_state;
}
//-------------------------------------------------------------------------------------------------------------------
// 函数简介 SCC8660 摄像头初始化
// 参数说明 void
// 返回参数 uint8 1-失败 0-成功
// 使用示例 scc8660_init();
// 备注信息
//-------------------------------------------------------------------------------------------------------------------
uint8 scc8660_init (void)
{
uint8 loop_count = 0;
uint8 return_state = 0;
uint8 *receiver_buffer = NULL;
soft_iic_info_struct scc8660_iic_struct;
const int16 scc8660_set_confing_buffer[SCC8660_CONFIG_FINISH][2]=
{
{SCC8660_INIT, 0}, // 摄像头开始初始化
{SCC8660_AUTO_EXP, SCC8660_AUTO_EXP_DEF}, // 自动曝光
{SCC8660_BRIGHT, SCC8660_BRIGHT_DEF}, // 亮度设置
{SCC8660_FPS, SCC8660_FPS_DEF}, // 图像帧率
{SCC8660_SET_COL, SCC8660_W}, // 图像列数
{SCC8660_SET_ROW, SCC8660_H}, // 图像行数
{SCC8660_PCLK_DIV, SCC8660_PCLK_DIV_DEF}, // PCLK分频系数
{SCC8660_PCLK_MODE, SCC8660_PCLK_MODE_DEF}, // PCLK模式
{SCC8660_COLOR_MODE, SCC8660_COLOR_MODE_DEF}, // 图像色彩模式
{SCC8660_DATA_FORMAT, SCC8660_DATA_FORMAT_DEF}, // 输出数据格式
{SCC8660_MANUAL_WB, SCC8660_MANUAL_WB_DEF} // 手动白平衡
};
int16 scc8660_get_confing_buffer[SCC8660_CONFIG_FINISH - 1][2]=
{
{SCC8660_AUTO_EXP, 0},
{SCC8660_BRIGHT, 0}, // 亮度设置
{SCC8660_FPS, 0}, // 图像帧率
{SCC8660_SET_COL, 0}, // 图像列数
{SCC8660_SET_ROW, 0}, // 图像行数
{SCC8660_PCLK_DIV, 0}, // PCLK分频系数
{SCC8660_PCLK_MODE, 0}, // PCLK模式
{SCC8660_COLOR_MODE, 0}, // 图像色彩模式
{SCC8660_DATA_FORMAT, 0}, // 输出数据格式
{SCC8660_MANUAL_WB, 0}, // 白平衡设置
};
do
{
scc8660_type = SCC8660_SCCB; // 先尝试 SCCB 方式通信
soft_iic_init(&scc8660_iic_struct, // 配置 SCCB
0, // 地址暂设为 0
SCC8660_COF_IIC_DELAY, // 时钟延迟
SCC8660_COF_IIC_SCL, // SCL 引脚
SCC8660_COF_IIC_SDA); // SDA 引脚
if(!scc8660_sccb_check_id(&scc8660_iic_struct)) // 尝试获取 ID
{
set_camera_type(CAMERA_COLOR, NULL, scc8660_dvp_handler);
return_state = scc8660_sccb_set_config(scc8660_set_confing_buffer); // 获取到 ID 则进行参数配置
}
else // 没有获取到 ID 则进入下一步
{
scc8660_type = SCC8660_UART; // 切换到 UART 方式通信
set_camera_type(CAMERA_COLOR, scc8660_uart_handler, scc8660_dvp_handler);
receiver_buffer = (uint8 *)malloc(SCC8660_COF_BUFFER_SIZE); // 申请串口通信用的缓冲区
if(NULL == receiver_buffer) // 堆空间不足
{
return_state = 2; // 异常码
break;
}
camera_receiver_fifo = (fifo_struct *)malloc(sizeof(fifo_struct)); // 申请 FIFO 对象空间
if(NULL == camera_receiver_fifo) // 堆空间不足
{
return_state = 2; // 异常码
break;
}
fifo_init(camera_receiver_fifo, // 用申请的空间配置 FIFO
FIFO_DATA_8BIT, // 数据位宽 8bit
receiver_buffer, // 指向申请来的缓冲区
SCC8660_COF_BUFFER_SIZE); // 缓冲区大小
// 初始换串口 配置摄像头
uart_init(SCC8660_COF_UART, // 指定 SCC8660 的串口号
SCC8660_COF_BAUR, // 通信波特率
SCC8660_COF_UART_RX, // 芯片 TX 摄像头模块 RX
SCC8660_COF_UART_TX); // 芯片 RX 摄像头模块 TX
uart_rx_interrupt(SCC8660_COF_UART, 1); // 开启接收中断
if(scc8660_uart_check_id()) // 通过 UART 获取摄像头的 ID
{
uart_rx_interrupt(SCC8660_COF_UART, 0); // 摄像头的类型 ID 不正确 关闭串口接收中断
return_state = 3; // 检查一下是不是选错摄像头了
break;
}
do
{
if(scc8660_set_config(scc8660_set_confing_buffer)) // 通过 UART 发送配置参数
{
uart_rx_interrupt(SCC8660_COF_UART, 0); // 串口通信出错并超时退出了 关闭串口接收中断
return_state = 1; // 检查一下接线有没有问题 如果没问题可能就是坏了
break;
}
if(scc8660_get_config(scc8660_get_confing_buffer)) // 通过 UART 回读配置参数 查看配置是否正确
{
uart_rx_interrupt(SCC8660_COF_UART, 0); // 串口通信出错并超时退出了 关闭串口接收中断
return_state = 1; // 检查一下接线有没有问题 如果没问题可能就是坏了
break;
}
// 对比发送缓冲检查数据 如果数据正确则配置成功
for(loop_count = 0; SCC8660_CONFIG_FINISH - 1 > loop_count; loop_count ++)
{
if( scc8660_set_confing_buffer[loop_count + 1][0] != scc8660_get_confing_buffer[loop_count][0] ||
scc8660_set_confing_buffer[loop_count + 1][1] != scc8660_get_confing_buffer[loop_count][1])
{
break;
}
}
return_state = (SCC8660_CONFIG_FINISH - 1 == loop_count) ? 0 : 1;
}while(0);
}
if(!return_state)
{
// DVP引脚初始化
dvp_gpio_init(
SCC8660_D0_PIN, SCC8660_D1_PIN, SCC8660_D2_PIN, SCC8660_D3_PIN,
SCC8660_D4_PIN, SCC8660_D5_PIN, SCC8660_D6_PIN, SCC8660_D7_PIN,
SCC8660_PCLK_PIN, SCC8660_VSY_PIN, SCC8660_HERF_PIN);
// DVP接口初始化
dvp_camera_init((uint32 *)&scc8660_image[0], (uint32 *)&scc8660_image[0], SCC8660_IMAGE_SIZE, SCC8660_H);
break;
}
set_camera_type(NO_CAMERE, NULL, NULL);
}while(0);
free(receiver_buffer); // 释放缓冲器内存
free(camera_receiver_fifo); // 释放 FIFO 内存
camera_receiver_fifo = NULL; // FIFO 指针指向 NULL
return return_state;
}