BCU/library/bsp/bsp_bf8915a.h

387 lines
16 KiB
C
Raw Permalink 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.

/**
******************************************************************************
* @file BF8915A.h
* @version V1.1.4
* @date 2021-12-20
* @brief This file provides the BF8915A functions(该文件提供BF8915A功能).
*/
#ifndef __BF8915A_H__
#define __BF8915A_H__
//#include "kit_sys.h"
#include <stdlib.h>
#include "stdint.h"
#include <stdio.h>
#include "gpio_manager.h"
typedef enum
{
FALSE = 0U,
TRUE = 1U,
}BOOLEAN_Type;
typedef unsigned char BOOLEAN;
//typedef BOOLEAN_Type BOOLEAN;//布尔类型的值代替了stdbool中定义的bool值方便客户做移植
extern BOOLEAN isospi_reverse;//改变该变量可以实现菊花链SPI正反向通信。
//FALSE:相对的正向通信本SDK中的SPI1为相对正向通信SPI1菊花链连接的第一颗IC看作是IC0;
//TURE: 相对的反向通信本SDK中的SPI2为相对反向通信但也是SPI1菊花链连接的第一颗IC看作是IC0。
extern const uint16_t crc15Table[256];//precomputed CRC15 Table(预计算15位循环冗余校验值的表)
#define TOTAL_IC 100U //菊花链上串接IC(8915A)的个数,需根据实际情况来修改此数
#define CELL_CHANNELS 16U //16个电池通道
#define GPIO_CHANNELS 9U //8个GPIO + 1个VD33T
#define STAT_CHANNELS 8U //8个16位的电压数据
#define NUM_CV_REG 6U //Cell Voltage Register Group(电池电压寄存器组) A B C D E F共6个
#define NUM_GOIO_REG 3U //Auxiliary Register Group(含有GPIOx电压信息)A B C 共3个
#define NUM_STAT_REG 4U //Status Register Group(状态寄存器组) A B C D共4个
//ADC转换的电池选择
#define CSEL_CH_ALL 0U
#define CSEL_CH_1and9 1U
#define CSEL_CH_2and10 2U
#define CSEL_CH_3and11 3U
#define CSEL_CH_4and12 4U
#define CSEL_CH_5and13 5U
#define CSEL_CH_6and14 6U
#define CSEL_CH_7and15 7U
#define CSEL_CH_8and16 8U
//ADC转换的GPIO选择
#define GSEL_CH_ALL 0U //所有GPIO0~GPIO7,VD33T
#define GSEL_CH_GPIO0 1U
#define GSEL_CH_GPIO1 2U
#define GSEL_CH_GPIO2 3U
#define GSEL_CH_GPIO3 4U
#define GSEL_CH_VD33T 5U
#define GSEL_CH_GPIO4 6U
#define GSEL_CH_GPIO5 7U
#define GSEL_CH_GPIO6 8U
#define GSEL_CH_GPIO7 9U
//ADC转换内部参数选择 STSEL
#define STSEL_CH_ALL 0U //所有内部参数VD33、VH、VD7、VD5、VD4、VTEMP、VREF2、VREF_LP
#define STSEL_CH_VD33 1U
#define STSEL_CH_VH 2U
#define STSEL_CH_VD7 3U
#define STSEL_CH_VD5 4U
#define STSEL_CH_VD4 5U
#define STSEL_CH_VTEMP 6U
#define STSEL_CH_VREF2 7U
#define STSEL_CH_VREF_LP 8U
//ADC频率选择
#define CLKSEL_2M 1U
#define CLKSEL_4M 0U //ADCW过采样率为4096(ADCOW指令时使用,其他指令请勿使用)
//CFGAR2
#define OSR_SEL FALSE //0U ADC过采样率为64、128、256、512(四种过采样率通过指令的OSR选择)
#define OSR_SEL_DISABLE TRUE //1U ADCW过采样率为4096(ADCOW指令时使用其他指令请勿使用)
//ADC过采样率选择
#define OSR_64 0U
#define OSR_128 1U
#define OSR_256 2U
#define OSR_512_4096 3U
//放电允许
#define DISCP_DISABLE 0U //DISCP=0当对应的电池和相邻的电池被测量时S脚的放电状态将不使能
#define DISCP_ENABLE 1U //DISCP=1在电池测量期间S脚的放电状态不改变
//自测试模式选择
#define SELFTEST_6666 0U //ST = 0,输出样本0x6666
#define SELFTEST_9999 1U //ST = 1,输出样本0x9999
//检测模式选择
#define MNMOD_DISABLE 0U //不进监测模式,指令无效
#define MNMOD_1 1U //监测模式1,模拟电源和时钟打开pll_100m打开
#define MNMOD_2 2U //监测模式2,模拟电源和时钟打开pll_100m关闭
#define MNMOD_3 3U //监测模式3,低功耗监测,模拟电源和时钟关闭
//检测模式开关
#define STRMN_ENABLE 1U //开启监测模式
#define STRMN_DISABLE 0U //关闭监测模式
//GPIO断线检测电流源选择
#define DOWN_GPIO_UP 0U //上拉电流源
#define DOWN_GPIO_DOWN 1U //下拉电流源
//均衡放电电池奇偶选择
#define OESEL_ODD 0U //奇数节
#define OESEL_EVEN 1U //偶数节
//均衡开关总控制信号2
#define BLEN_ON 1U //开启
#define BLEN_OFF 0U //关闭
/***命令的bit[12:9]位***/
#define CMD_ADCV 0x01U
#define CMD_ADCOW 0x02U
#define CMD_CVST 0x03U
#define CMD_ADOL 0x04U
#define CMD_ADGP 0x05U
#define CMD_ADGOW 0x06U
#define CMD_GPST 0x07U
#define CMD_ADSTAT 0x08U
#define CMD_STATST 0x09U
#define CMD_ADCVGP 0x0aU
#define CMD_ADCVVH 0x0bU
#define CMD_MNT 0x0cU
#define CMD_STRBL 0x0dU
/***配置寄存器***/
#define RDCFGA 0x05U
#define RDCFGB 0x06U
#define RDCFGC 0x07U
#define RDCFGD 0x08U
/***电池电压寄存器对应的命令***/
#define CVAR 0x01U // 0x01 + 0x08 = 0x09U 函数BF8915A_rdcv_reg中用到
#define CVBR 0x02U // 0x02 + 0x08 = 0x0AU
#define CVCR 0x03U // 0x03 + 0x08 = 0x0BU
#define CVDR 0x04U // 0x04 + 0x08 = 0x0CU
#define CVER 0x05U // 0x05 + 0x08 = 0x0DU
#define CVFR 0x06U // 0x06 + 0x08 = 0x0EU
/***GPIOG电压***/
#define GVAR 0x01U // 0x01 + 0x0E = 0x0FU 函数BF8915A_rdgv_reg中用到
#define GVBR 0x02U // 0x02 + 0x0E = 0x10U
#define GVCR 0x03U // 0x03 + 0x0E = 0x11U
/***状态寄存器RDSTAT***/
#define STATA 0x01U // 0x01 + 0x11 = 0x12U
#define STATB 0x02U // 0x02 + 0x11 = 0x13U
#define STATC 0x03U // 0x03 + 0x11 = 0x14U
#define STATD 0x04U // 0x04 + 0x11 = 0x15U
#define REG_ALL 0x00U
#define NUM_RX_BYT 0x08U //接收到数据的字节数6个数据+2个PEC(检错码)
typedef struct//Register data structure(寄存器数据结构).
{
uint8_t tx_data[6]; //由于每个寄存器组有六byte(字节)所以数组中元素个数为6
uint8_t rx_data[8]; //接收到的寄存器组中 6 byte(字节)+ 2 byte PEC(检错码)
uint8_t rx_pec_match; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}Reg_TypeDef;
typedef struct//ConfigA data structure(配置A数据结构).
{
uint8_t gpio_pd_en; //GPIO下拉使能每一位对应一个IO口
uint8_t gpio_analog_en; //GPIO0~7模式选择 0:模拟输入 1:普通GPIO
BOOLEAN adc_init_mode;
BOOLEAN rsvd2; //保留位永远置0
BOOLEAN rsvd1; //保留位永远置0
BOOLEAN osr_sel; //ADC过采样率选择 1:ADC过采样率为1024、2048、4096、4096 0:ADC过采样率为64、128、256、512
BOOLEAN refon; //VREF1(基准电压)配置 1 :VREF1保持上电直到WDT溢出 0 :ADC转换完之后VREF1关闭 (默认)
uint16_t uv_th; //欠压阈值
uint16_t ov_th; //过压阈值
uint8_t rx_pec_match; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}ConfigA_TypeDef;
typedef struct//ConfigB data structure(配置B数据结构).
{
uint16_t dcc; //放电电池开关1:打开Cell x 的开关 0:关闭Cell x 的开关 (默认)
uint8_t bl_duty_sel; //主动均衡占空比选择 占空比=(BL_DUTY_SEL+1)*10%配置范围10%~100%当大于9时是100%。
uint8_t dcto; //放电定时溢出值配置
uint16_t vuv_bl_th; //均衡欠压阈值
BOOLEAN dtmen; //在自动均衡下如果DTMEN=1当欠压(小于VUV_BL)关闭放电开关如果DTMEN=0不进行欠压比较
uint8_t rx_pec_match; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}ConfigB_TypeDef;
typedef struct//ConfigC data structure(配置C数据结构).
{
uint16_t vuv_m_th; //监测模式下,欠压阈值
uint16_t vov_m_th; //监测模式下,过压阈值
uint16_t itov_m_th; //监测模式下,内部过温阈值
uint16_t extov_m_th; //监测模式下,外部过温阈值
uint8_t rx_pec_match; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}ConfigC_TypeDef;
typedef struct//ConfigD data structure(配置D数据结构).
{
uint16_t csel_m; //监测模式下,电池电压监测比较选择 1:相应的电池就会在监测模式下进行过欠压比较,置标志位
// 0:相应的电池就会在监测模式下不进行过欠压比较,不置标志位
uint8_t gsel_m; //监测模式下GPIO0~7比较选择
BOOLEAN isospi_wake_t; //isospi(菊花链SPI)唤醒时间0:10us 1:20us
BOOLEAN adc_rst_sel; //测量第1个通道ADC复位选择0:所有测量通道都复位ADC1:只有第1个通道复位ADC之后2~n通道不复位ADC
uint8_t time_sel_m; //监测模式下,定时监测时间配置,定时时间=(TIME_SEL_M+1)*1s(2~64s)最小配置值是1(2s)
uint8_t adc_init_t2; //ADC测量第2~n个通道ADC初始化时间(ADC_INIT_T2+1)*20us(20~320us)
uint8_t adc_init_t1; //ADC测量第1个通道ADC初始化时间(ADC_INIT_T1+1)*100us(100~1600us)
BOOLEAN trim2_en; //修调2使能1:使能0:不使能
BOOLEAN trim1_en; //修调1使能1:使能0:不使能
uint8_t err_t_sel_m; //监测模式下发生异常,上报的间隔时间 00:200ms01:400ms10:600ms11:800ms
uint8_t vd33t_on_m; //监测模式下VREF1和VD33T的启动时间配置(5+VD33T_ON_M*10)ms范围是5~155ms
uint8_t rx_pec_match; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}ConfigD_TypeDef;
typedef struct//Cell Voltage data structure(电池电压数据结构).
{
//因为有16节电池所以数组的有效数为16个
uint16_t cellVoltage[16];
//因为有ABCDEF六个电池电压寄存器
uint8_t pec_match[6]; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}CellVoltage_TypeDef;
typedef struct//AUX Reg Voltage Data structure(GPIO寄存器电压数据结构)
{
uint16_t gpioVoltage[9]; //GPIO Voltage Codes(辅助寄存器中的G0V~G7V)GPIOA、B、C(8个GPIO + 1个VD33T)
uint8_t pec_match[3]; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}GPIOReg_TypeDef;
typedef struct//Status Reg data structure(状态寄存器数据结构)
{
uint16_t stat_codes[8]; //A two dimensional array of the stat voltage codes(统计电压代码的二维数组)
uint8_t uvOvFlags[4]; //byte array that contains the uv/ov flag data(包含uv/ov标志数据的字节数组)
uint16_t cellUvFlag; //电池欠压标志位,每个位用来标志每个电池的欠压标志位
uint16_t cellOvFlag; //电池过压标志位,每个位用来标志每个电池的过压标志位
uint8_t cfg_err; //配置字校验出错标志(包括奇偶异或校验和ECC校验)
uint8_t pec_err; //指令数据的PEC校验出错标志
uint8_t extov; //监测模式下,外部温度(gpio)过温标志
uint8_t itov; //监测模式下,内部温度(vtemp)过温标志
uint8_t htov2; //硬件温度(95℃)过温标志, 1:硬件温度过温0:硬件温度不过温
uint8_t htov1; //硬件温度(125℃)过温标志1:硬件温度过温0:硬件温度不过温
uint8_t mux_fail; //开关译码测试结果, 0 : 开关译码正确 1: 开关译码错误
uint8_t mode_state; //模式状态寄存器, 00:standby模式 01:测量模式 10:监测模式 11:保留位(sleep模式关闭时钟不能读出相关状态)
uint8_t rst_state; //软件复位标志寄存器
uint8_t pec_match[4]; //如果在最近的读取cmd(指令)期间检测到PEC(检错码)错误
}StateReg_TypeDef; //A、B、C、D四个状态寄存器
/***************************需要修改的函数********************************************
*其中包括
*片选CS :相对正向:cs_low()和cs_high();相对反向:cs_low_reverse()和cs_high_reverse()
*延时函数 :delay_us
*SPI接口 :SPI1_ReadWriteByte和SPI2_ReadWriteByte
*********************************END************************************************/
#define cs_low() drv_gpio_set_pin_status(kGpioType_SPI1_Cs, kGpioStatus_Low);//将相对正向通信的片选CS设置为低电平
#define cs_high() drv_gpio_set_pin_status(kGpioType_SPI1_Cs, kGpioStatus_High)//将相对正向通信的片选CS设置为高电平
#define cs_low_reverse() drv_gpio_set_pin_status(kGpioType_SPI2_Cs, kGpioStatus_Low);//{PDout(8) = 0U;}//将相对反向通信的片选CS设置为低电平
#define cs_high_reverse() drv_gpio_set_pin_status(kGpioType_SPI2_Cs, kGpioStatus_High);//将相对反向通信的片选CS设置为高电平
//延时函数
extern void delay_us(uint32_t nus);
//TxData 可以是8位或16位的在启用SPI之前就确定好数据帧格式
extern uint8_t SPI1_ReadWriteByte(uint8_t TxData);
//TxData 可以是8位或16位的在启用SPI之前就确定好数据帧格式
extern uint8_t SPI2_ReadWriteByte(uint8_t TxData);
//Wake isoSPI up from idle state(将菊花链SPI从空闲状态唤醒)
void wakeup_idle(uint8_t total_ic);
///Generic wakeup command to wake the BF8915A from sleep(将BF8915A从睡眠中唤醒的通用唤醒命令)
void wakeup_sleep(uint8_t total_ic);
//Writes an array of bytes out of the SPI port(将字节数组写入SPI端口)
void spi_write_array(uint8_t len,uint8_t writeData[]);
//Generic function to write BF8915A commands. Function calculated PEC for tx_cmd data(编写BF8915A命令的通用函数。函数为cmd数据计算了检错码)
void BF8915A_cmd(uint8_t tx_cmd[2]);
//Generic function to write BF8915A commands and write payload data. Function calculated PEC for tx_cmd data(写BF8915A命令和写有效载荷数据的通用函数。tx_cmd数据的函数计算检错码)
void BF8915A_write(uint8_t total_ic ,uint8_t tx_cmd[2],uint8_t write_data[]);//Writes an array of data to the daisy chain (将一组数据写入菊花链)
//Issues a command onto the daisy chain and reads back 6*total_ic data in the rx_data array(向菊花链发出命令回读rx_data数组中的6倍IC(8915A)数据)
uint8_t BF8915A_read(uint8_t total_ic,uint8_t tx_cmd[2],uint8_t *rx_data);
//Calculates and returns the CRC15(计算并返回15位循环冗余校验值)
uint16_t pec15_calc(uint8_t len,uint8_t *pecData);
//Write the BF8915A CFGRA(写BF8915A的配置寄存器组A)
void BF8915A_wrcfga(uint8_t total_ic,ConfigA_TypeDef configAData[]);
//Write the BF8915A CFGRB(写BF8915A的配置寄存器组B)
void BF8915A_wrcfgb(uint8_t total_ic,ConfigB_TypeDef configBData[]);
//Write the BF8915A CFGRC(写BF8915A的配置寄存器组C)
void BF8915A_wrcfgc(uint8_t total_ic,ConfigC_TypeDef configCData[]);
//Write the BF8915A CFGRD(写BF8915A的配置寄存器组D)
void BF8915A_wrcfgd(uint8_t total_ic,ConfigD_TypeDef configDData[]);
//Reads the BF8915A CFGRA register(读BF8915A的配置寄存器组A)
uint8_t BF8915A_rdcfga(uint8_t total_ic,ConfigA_TypeDef configAData[]);
//Reads the BF8915A CFGRB register(读BF8915A的配置寄存器组B)
uint8_t BF8915A_rdcfgb(uint8_t total_ic,ConfigB_TypeDef configBData[]);
//Reads the BF8915A CFGRC register(读BF8915A的配置寄存器组C)
uint8_t BF8915A_rdcfgc(uint8_t total_ic,ConfigC_TypeDef configCData[]);
//Reads the BF8915A CFGRD register(读BF8915A的配置寄存器组D)
uint8_t BF8915A_rdcfgd(uint8_t total_ic,ConfigD_TypeDef configDData[]);
// Reads the raw cell voltage register data(读取原始电池电压寄存器数据)
void BF8915A_rdcv_reg(uint8_t reg,uint8_t total_ic,uint8_t *cellData);
//reads BF8915A GPIO registers(读取BF8915A的GPIO寄存器)
void BF8915A_rdgv_reg(uint8_t reg,uint8_t total_ic,uint8_t *gpioData);
//Reads BF8915A stat registers(读取BF8915A状态寄存器)
void BF8915A_rdstat_reg(uint8_t reg,uint8_t total_ic,uint8_t *statData);
//ADCV CMD(电池电压ADC转换指令)
void BF8915A_adcv(uint8_t CLKSEL,uint8_t OSR,uint8_t DISCP,uint8_t CSEL);
//ADCOW CMD(电池断线电压ADC转换指令)
void BF8915A_adcow(uint8_t CLKSEL,uint8_t OSR,uint8_t DISCP,uint8_t CSEL);
//CVST CMD(电池电压自测试命令)
void BF8915A_cvst(uint8_t CLKSEL,uint8_t OSR,uint8_t SFTEST);
//ADOL CMD(电池重叠单元测量指令)
void BF8915A_adol(uint8_t CLKSEL,uint8_t OSR,uint8_t DISCP);
//ADGP CMD(GPIO ADC转换指令)
void BF8915A_adgp(uint8_t CLKSEL,uint8_t OSR,uint8_t GSEL);
//GPST CMD(GPIO输入自测试指令)
void BF8915A_gpst(uint8_t CLKSEL,uint8_t OSR,uint8_t SFTEST);
//ADSTAT CMD(测量内部设备参数指令)
void BF8915A_adstat(uint8_t CLKSEL,uint8_t OSR,uint8_t STSEL);
//STATST CMD(自测试内部设备参数指令)
void BF8915A_statst(uint8_t CLKSEL,uint8_t OSR,uint8_t SFTEST);
//ADCVGP CMD(测量电池电压和GPIO输入指令)
void BF8915A_adcvgp(uint8_t CLKSEL,uint8_t OSR,uint8_t DISCP);
//MNT CMD(监测指令)
void BF8915A_mnt(uint8_t CLKSEL, uint8_t OSR,uint8_t MNMOD,uint8_t STRMN,uint8_t BLEN);
//STRBL CMD(被动均衡指令)
void BF8915A_strbl(uint8_t CLKSEL,uint8_t OSR,uint8_t OESEL,uint8_t BLEN);
//Clear Cell Register(清除单元寄存器)
void BF8915A_clrcell(void);
//Clear GPIO Register(清除GPIO寄存器)
void BF8915A_clrgp(void);
//Clear STATE Register(清除状态寄存器)
void BF8915A_clrstat(void);
//PLADC CMD(轮询ADC转换状态指令)
uint8_t BF8915A_pladc(void);
//soft reset(软件复位)
void BF8915A_softrst(void);
//Helper function that parses voltage measurement registers(解析电压测量寄存器的辅助函数)
uint8_t parse_cells(uint8_t current_ic,uint8_t cell_reg,uint8_t cell_data[],uint16_t *cell_codes,uint8_t *ic_pec);
//Reads and parses the BF8915A cell voltage registers(读取和解析BF8915A电池电压寄存器)
uint8_t BF8915A_rdcv(uint8_t reg,uint8_t total_ic,CellVoltage_TypeDef *cellVoltage);
//Reads and parses the BF8915A GPIO registers(读取和解析BF8915A的GPIO寄存器)
uint8_t BF8915A_rdgv(uint8_t reg,uint8_t total_ic,GPIOReg_TypeDef *gpioVoltage);
//Reads and parses the BF8915A stat registers(读取和解析BF8915A状态寄存器)
uint8_t BF8915A_rdstat(uint8_t reg,uint8_t total_ic,StateReg_TypeDef *stateReg);
#endif