This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

[参考译文] BQ79656-Q1:CRO 上未接收电芯电压数据/未接收数据

Guru**** 2589300 points
Other Parts Discussed in Thread: BQ79656-Q1, BQ79616-Q1, BQ79616

请注意,本文内容源自机器翻译,可能存在语法或其它翻译错误,仅供参考。如需获取准确内容,请参阅链接中的英语原文或自行翻译。

https://e2e.ti.com/support/power-management-group/power-management/f/power-management-forum/1321346/bq79656-q1-cell-voltage-data-not-receive-no-data-receive-on-cro

器件型号:BQ79656-Q1
主题中讨论的其他器件: BQ79616-Q1BQ79616

/***********************************************************************************************************************
* DISCLAIMER
* This software is supplied by Renesas Electronics Corporation and is only intended for use with Renesas products.
* No other uses are authorized. This software is owned by Renesas Electronics Corporation and is protected under all
* applicable laws, including copyright laws. 
* THIS SOFTWARE IS PROVIDED "AS IS" AND RENESAS MAKES NO WARRANTIES REGARDING THIS SOFTWARE, WHETHER EXPRESS, IMPLIED
* OR STATUTORY, INCLUDING BUT NOT LIMITED TO WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NON-INFRINGEMENT.  ALL SUCH WARRANTIES ARE EXPRESSLY DISCLAIMED.TO THE MAXIMUM EXTENT PERMITTED NOT PROHIBITED BY
* LAW, NEITHER RENESAS ELECTRONICS CORPORATION NOR ANY OF ITS AFFILIATED COMPANIES SHALL BE LIABLE FOR ANY DIRECT,
* INDIRECT, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES FOR ANY REASON RELATED TO THIS SOFTWARE, EVEN IF RENESAS OR
* ITS AFFILIATES HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
* Renesas reserves the right, without notice, to make changes to this software and to discontinue the availability 
* of this software. By using this software, you agree to the additional terms and conditions found by accessing the 
* following link:
* www.renesas.com/disclaimer
*
* Copyright (C) 2012, 2021 Renesas Electronics Corporation. All rights reserved.
***********************************************************************************************************************/

/***********************************************************************************************************************
* File Name    : r_main.c
* Version      : CodeGenerator for RL78/F13 V2.03.07.02 [08 Nov 2021]
* Device(s)    : R5F10BGG
* Tool-Chain   : CCRL
* Description  : This file implements main function.
* Creation Date: 2024-01-30
***********************************************************************************************************************/

/***********************************************************************************************************************
Includes
***********************************************************************************************************************/
#include "r_cg_macrodriver.h"
#include "r_cg_cgc.h"
#include "r_cg_port.h"
#include "r_cg_serial.h"
#include "r_cg_timer.h"
/* Start user code for include. Do not edit comment generated here */
#include "Defines.h"
#include <stdio.h>
#include <stdlib.h>
#include "math.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "stdbool.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include <stdint.h>//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "string.h"
#include <math.h>
#include <stdio.h>
//#include "A0_reg.h"
#include "B0_reg.h"
#include "datatypes.h"
#include "bq79616.h"
#include "datatypes.h"
/* End user code. Do not edit comment generated here */
#include "r_cg_userdefine.h"

/***********************************************************************************************************************
Pragma directive
***********************************************************************************************************************/
/* Start user code for pragma. Do not edit comment generated here */

/* End user code. Do not edit comment generated here */

/***********************************************************************************************************************
Global variables and functions
***********************************************************************************************************************/
/* Start user code for global. Do not edit comment generated here */
int i ;extern int currentBoard;
int boardByteStart;
uint16_t rawData;
float cellVoltage;
uint8_t Tx_Buffer[7]={0x90,0x00,0x03,0x0A,0X01,0XD3,0X7D};//WRITE FOR ON THE TREFERENCE 
//uint8_t Tx_Buffer1[7]={0x80,0x00,0x05,0x87,0X00,0X56,0X2F};//FOR READ CELL1 LO VOLTAGE
uint8_t Tx_Buffer1[6]={0xC0,0x00,0x05,0x87,0XEB,0XD8};//FOR READ CELL1 LO VOLTAGE
uint8_t Tx_Buffer2[6]={0xC0,0x05,0x05,0x87,0XBE,0XD8};//FOR READ CELL1 LO VOLTAGE
unsigned char UART1_RX_BUFFER[7],BytesReceived;
//for bq communication
void uSDelay(unsigned int delay_time);
void mSDelay(unsigned int delay_time1);
uint16_t count_us;
void configurePinAsGPIO(void);
void configurePinAsUART(void);
void TestForPinAlternateFunctionality(void);
/* End user code. Do not edit comment generated here */
void R_MAIN_UserInit(void);

/***********************************************************************************************************************
* Function Name: main
* Description  : This function implements main function.
* Arguments    : None
* Return Value : None
***********************************************************************************************************************/
void main(void)
{
    R_MAIN_UserInit();
    /* Start user code. Do not edit comment generated here */
    //P0 = P0 ^ (1<<0);
    //for bq communication
    UART_RX_RDY = 0;
    // Initialize the system and other configurations as needed
      
    //Your application code here

    // Your UART communication code here
    //INITIALIZE BQ79616-Q1
    //need 2 wakes as this particular microcontroller outputs RX=0 by default, and so puts devices into hardware reset while the program is being loaded
    Wake79616();
    mSDelay(12*TOTALBOARDS); //wake tone duration is ~1.6ms per board + 10ms per board for each device to wake up from shutdown = 11.6ms per 616 board (rounded to 12ms since variable used is an integer)
    Wake79616();
    mSDelay(12*TOTALBOARDS); //wake tone duration is ~1.6ms per board + 10ms per board for each device to wake up from shutdown = 11.6ms per 616 board (rounded to 12ms since variable used is an integer)
 
    AutoAddress();
    uSDelay(4000); //4ms total required after shutdown to wake transition for AFE settling time, this is for top device only
    WriteReg(0, FAULT_MSK2, 0x40, 1, FRMWRT_ALL_W); //MASK CUST_CRC SO CONFIG CHANGES DON'T FLAG A FAULT//broadcast WRITE
    ResetAllFaults(0, FRMWRT_ALL_W); //CLEAR ALL FAULTS//broadcast WRITE	
    
     //VARIABLES
     currentBoard = 0;

    //ARRAYS (MUST place out of loop so not re-allocated every iteration)
    //BYTE response_frame[(16*2+6)*TOTALBOARDS]; //hold all 16 vcell*_hi/lo values

    //OPTIONAL EXAMPLE FUNCTIONS
    //RunCB();
    //ReverseAddressing();
    
    //set up the main ADC
    WriteReg(0, ACTIVE_CELL, 0x0A, 1, FRMWRT_ALL_W);    //set all cells to active
    WriteReg(0, ADC_CONF1, 0x02, 1, FRMWRT_ALL_W);      //26Hz LPF_Vcell (38ms average)
    WriteReg(0, ADC_CTRL1, 0x0E, 1, FRMWRT_ALL_W);      //continuous run, LPF enabled and MAIN_GO
    uSDelay(38000+5*TOTALBOARDS);                       //initial delay to allow LPF to average for 38ms (26Hz LPF setting used)
    //R_UART1_Send(Tx_Buffer1,7);
    while (1U)
    {
             //*******************
        //READ CELL VOLTAGES
        //*******************
        //reset variables
        //i = 0;
        //currentBoard = 0;

        //clear out the array so we know we have fresh data every loop
        memset(response_frame,0,sizeof(response_frame));

        //wait single round robin cycle time + reclocking delay for each device, every loop
        uSDelay(192+5*TOTALBOARDS);

        //read all the values (HI/LO for each cell = 32 total)
        ReadReg(0, VCELL16_HI, response_frame, 16*2, 0, FRMWRT_ALL_R);

        /*
         * ***********************************************
         * NOTE: SOME COMPUTERS HAVE ISSUES TRANSMITTING
         * A LARGE AMOUNT OF DATA VIA PRINTF STATEMENTS.
         * THE FOLLOWING PRINTOUT OF THE RESPONSE DATA
         * IS NOT GUARANTEED TO WORK ON ALL SYSTEMS.
         * ***********************************************
        */

        //format and print the resultant response frame
        printf("\n"); //start with a newline to add some extra spacing between loop
        for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
        {
            //response frame actually starts with top of stack, so currentBoard is actually inverted from what it should be
            printf("BOARD %d:\t",TOTALBOARDS-currentBoard);

            //go through each byte in the current board (32 bytes = 16 cells * 2 bytes each)
            for(i=0; i<32; i+=2)
            {
                //each board responds with 32 data bytes + 6 header bytes
                //so need to find the start of each board by doing that * currentBoard
                boardByteStart = (16*2+6)*currentBoard;

                //convert the two individual bytes of each cell into a single 16 bit data item (by bit shifting)
                rawData = (response_frame[boardByteStart+i+4] << 8) | response_frame[boardByteStart+i+5];

                //cast as a signed 16 bit data item, and multiply by 190.73uV to get an actual voltage
                cellVoltage = ((int16_t)rawData)*0.00019073;

                //print the voltages - it is (32-i)/2 because cells start from 16 down to 1
                //and there are 2 bytes per cell (i value is twice the cell number)
                printf("Cell %d = %f\t", (32-i)/2, cellVoltage);
            }
            printf("\n"); //newline per board
        }
        //**********************
        //END READ CELL VOLTAGES
        //**********************	    
    }
    /* End user code. Do not edit comment generated here */
}

/***********************************************************************************************************************
* Function Name: R_MAIN_UserInit
* Description  : This function adds user code before implementing main function.
* Arguments    : None
* Return Value : None
***********************************************************************************************************************/
void R_MAIN_UserInit(void)
{
    /* Start user code. Do not edit comment generated here */
    EI();
	
    /* End user code. Do not edit comment generated here */
}

/* Start user code for adding. Do not edit comment generated here */

void uSDelay(unsigned int delay_time)
 {
  count_us=0;
  R_TAU0_Channel0_Start();
  while(count_us < delay_time-1)
   {
    ;
   }
 }
 void mSDelay(unsigned int delay_time1)
 {
  unsigned int j;
  for (j=0; j<delay_time1; j++)
   {uSDelay(1000);}
 }
///////////////
void configurePinAsGPIO(void) 
 {
  // Set the corresponding GPIO pin as an output
  R_UART1_Stop();
  P1 = _00_Pn2_OUTPUT_0;//declare P12 AS I/O port PIN
  PSRSEL = PSRSEL | _00_PSR12_NORMAL;//SET P12 SLEW RATE NORMAL
  PM1 =  PM1 | _00_PMn2_MODE_OUTPUT;//SET PORT MODE AS OUTPUT PIN
 }
void configurePinAsUART(void) 
 {
  // Configure the pin for UART1 functionality
  R_SAU1_Create(); 
  R_UART1_Start();
 }
void TestForPinAlternateFunctionality(void)
 {
	//use uart tx pin as gpio/uart tx
	//Configure the GPIO pin as an output
	configurePinAsGPIO();
	//P1=P1|(1<<2);
	P1 = P1 ^ (1<<2);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2); 
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2);
	  
	// Change the pin functionality to UART
	configurePinAsUART();
	R_UART1_Send(Tx_Buffer,7);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000); 

	//Configure the GPIO pin as an output
	configurePinAsGPIO();
	P1 = P1 ^ (1<<2);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2); 
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	uSDelay(65000);uSDelay(65000);uSDelay(65000);uSDelay(65000);
	P1 = P1 ^ (1<<2);  

	// Change the pin functionality to UART	  
	configurePinAsUART();	  
 }
/* End user code. Do not edit comment generated here */
//bq79616.h
/*
 *  @file bq79616.h
 *
 *  @author Vince Toledo - Texas Instruments Inc.
 *  @date February 2020
 *  @version 1.2
 *  @note Built with CCS for Hercules Version: 8.1.0.00011
 *  @note Built for TMS570LS1224 (LAUNCH XL2)
 */

/*****************************************************************************
**
**  Copyright (c) 2011-2019 Texas Instruments
**
******************************************************************************/


#ifndef BQ79616_H_
#define BQ79616_H_

#include "datatypes.h"
//#include "hal_stdtypes.h"
#include <stdio.h>
#include <stdlib.h>
#include "math.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "stdbool.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include <stdint.h>//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "string.h"
#include <math.h>
#include <stdio.h>

//****************************************************
// ***Register defines, choose one of the following***
// ***based on your device silicon revision:       ***
//****************************************************
//#include "A0_reg.h"
#include "B0_reg.h"

// User defines
#define TOTALBOARDS 1       //boards in stack
#define MAXBYTES (16*2)     //maximum number of bytes to be read from the devices (for array creation)
#define BAUDRATE 1000000    //device + uC baudrate

#define FRMWRT_SGL_R	0x00    //single device READ
#define FRMWRT_SGL_W	0x10    //single device WRITE
#define FRMWRT_STK_R	0x20    //stack READ
#define FRMWRT_STK_W	0x30    //stack WRITE
#define FRMWRT_ALL_R	0x40    //broadcast READ
#define FRMWRT_ALL_W	0x50    //broadcast WRITE
#define FRMWRT_REV_ALL_W 0x60   //broadcast WRITE reverse direction
                  
// Function Prototypes
void Wake79616();
void AutoAddress();
BOOL GetFaultStat();

uint16_t CRC16(BYTE *pBuf, int nLen);

int  WriteReg(BYTE bID, uint16_t wAddr, uint64_t dwData, BYTE bLen, BYTE bWriteType);
int  ReadReg(BYTE bID, uint16_t wAddr, BYTE * pData, BYTE bLen, uint32_t dwTimeOut, BYTE bWriteType);

int  WriteFrame(BYTE bID, uint16_t wAddr, BYTE * pData, BYTE bLen, BYTE bWriteType);
int  ReadFrameReq(BYTE bID, uint16_t wAddr, BYTE bByteToReturn,BYTE bWriteType);
int  WaitRespFrame(BYTE *pFrame, uint32_t bLen, uint32_t dwTimeOut);

void delayms(uint16_t ms);
void delayus(uint16_t us);

void ResetAllFaults(BYTE bID, BYTE bWriteType);
void MaskAllFaults(BYTE bID, BYTE bWriteType);

void RunCB();
void ReverseAddressing();

#endif /* BQ79606_H_ */
//EOF
//bq79616.c
/*
 *  @file bq79616.c
 *
 *  @author Vince Toledo - Texas Instruments Inc.
 *  @date February 2020
 *  @version 1.2
 *  @note Built with CCS for Hercules Version: 8.1.0.00011
 *  @note Built for TMS570LS1224 (LAUNCH XL2)
 */

/*****************************************************************************
 **
 **  Copyright (c) 2011-2019 Texas Instruments
 **
 ******************************************************************************/
#include "r_cg_macrodriver.h"
#include "r_cg_cgc.h"
#include "r_cg_port.h"
#include "r_cg_serial.h"
#include "r_cg_timer.h"
#include <bq79616.h>
#include "string.h"
//#include "sci.h"
//#include "rti.h"
//#include "gio.h"
#include "datatypes.h"
#include <stdio.h>
#include <stdlib.h>
#include "math.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "stdbool.h"//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include <stdint.h>//$NO EFFECT ON HEX FILE SIZE IF COMMENTED
#include "string.h"
#include <math.h>
#include <stdio.h>
#include "Defines.h"



//GLOBAL VARIABLES (use these to avoid stack overflows by creating too many function variables)
//avoid creating variables/arrays in functions, or you will run out of stack space quickly
//BYTE response_frame2[(MAXBYTES+6)*TOTALBOARDS]; //response frame to be used by every read

extern BYTE fault_frame[39*TOTALBOARDS]; //hold fault frame if faults are printed
int currentBoard = 0;
int currentCell = 0;
BYTE bReturn = 0;
int bRes = 0;
//int count = 10000;
BYTE bBuf[8];
uint8_t pFrame[64];
uint16_t wCRC = 0;
uint16_t wCRC16 = 0;
int crc_i = 0;
static volatile unsigned int delayval = 0; //for delayms and delayus functions

extern int UART_RX_RDY;
//extern int RTI_TIMEOUT;

//PINGS
void Wake79616(void) //timing of wake ping at page no. 65 of datasheet
{
//    sciREG->GCR1 &= ~(1U << 7U); // put SCI into reset
//    sciREG->PIO0 &= ~(1U << 2U); // disable transmit function - now a GPIO
//    sciREG->PIO3 &= ~(1U << 2U); // set output to low

//    delayus(2500); // WAKE ping = 2ms to 2.5ms
//    sciInit();
//    sciSetBaudrate(sciREG, BAUDRATE);
	//P1 = P1 ^ (1<<2);
	configurePinAsGPIO();
	P1=P1|(1<<2);
	P1=P1&(~(1<<2));
	uSDelay(2250);
	P1=P1|(1<<2);
	configurePinAsUART();
}

void SD79616(void) {
//    sciREG->GCR1 &= ~(1U << 7U); // put SCI into reset
//    sciREG->PIO0 &= ~(1U << 2U); // disable transmit function - now a GPIO
//    sciREG->PIO3 &= ~(1U << 2U); // set output to low

//    delayus(9000); // SD(SHUTDOWN) ping = 7ms to 10ms
//    sciInit();
//    sciSetBaudrate(sciREG, BAUDRATE);
        configurePinAsGPIO();
	P1=P1|(1<<2);
	P1=P1&(~(1<<2));
	uSDelay(8500);
	P1=P1|(1<<2);
	configurePinAsUART();
}

void StA79616(void) {
//    sciREG->GCR1 &= ~(1U << 7U); // put SCI into reset
//    sciREG->PIO0 &= ~(1U << 2U); // disable transmit function - now a GPIO
//    sciREG->PIO3 &= ~(1U << 2U); // set output to low

//    delayus(250); // StA(SLEEP TO ACTIVE) ping = 250us to 300us
//    sciInit();
//    sciSetBaudrate(sciREG, BAUDRATE);
	configurePinAsGPIO();
        P1=P1|(1<<2);
	P1=P1&(~(1<<2));
	uSDelay(275);
	P1=P1|(1<<2);
	configurePinAsUART();
}

void HWRST79616(void) {//HARDWARE RESET 
//    sciREG->GCR1 &= ~(1U << 7U); // put SCI into reset
//    sciREG->PIO0 &= ~(1U << 2U); // disable transmit function - now a GPIO
//    sciREG->PIO3 &= ~(1U << 2U); // set output to low

//    delayus(36000); // StA ping = 36ms
//    sciInit();
//    sciSetBaudrate(sciREG, BAUDRATE);
        configurePinAsGPIO();
	P1=P1|(1<<2);
	P1=P1&(~(1<<2));
	uSDelay(36000);
	P1=P1|(1<<2);
	configurePinAsUART();
}
////END PINGS

//AUTO ADDRESS SEQUENCE
void AutoAddress()
{
    //DUMMY WRITE TO SNCHRONIZE ALL DAISY CHAIN DEVICES DLL (IF A DEVICE RESET OCCURED PRIOR TO THIS)
    WriteReg(0, OTP_ECC_DATAIN1, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN2, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN3, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN4, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN5, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN6, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN7, 0X00, 1, FRMWRT_ALL_W);
    WriteReg(0, OTP_ECC_DATAIN8, 0X00, 1, FRMWRT_ALL_W);

    //ENABLE AUTO ADDRESSING MODE
    WriteReg(0, CONTROL1, 0X01, 1, FRMWRT_ALL_W);

    //SET ADDRESSES FOR EVERY BOARD
    for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
    {
        WriteReg(0, DIR0_ADDR, currentBoard, 1, FRMWRT_ALL_W);
    }

    WriteReg(0, COMM_CTRL, 0x02, 1, FRMWRT_ALL_W); //set everything as a stack device first

    if(TOTALBOARDS==1) //if there's only 1 board, it's the base AND top of stack, so change it to those
    {
        WriteReg(0, COMM_CTRL, 0x01, 1, FRMWRT_SGL_W);
    }
    else //otherwise set the base and top of stack individually
    {
        WriteReg(0, COMM_CTRL, 0x00, 1, FRMWRT_SGL_W);
        WriteReg(TOTALBOARDS-1, COMM_CTRL, 0x03, 1, FRMWRT_SGL_W);
    }

    //SYNCRHONIZE THE DLL WITH A THROW-AWAY READ
    ReadReg(0, OTP_ECC_DATAIN1, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN2, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN3, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN4, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN5, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN6, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN7, response_frame2, 1, 0, FRMWRT_ALL_R);
    ReadReg(0, OTP_ECC_DATAIN8, response_frame2, 1, 0, FRMWRT_ALL_R);
//below commented already in original code of TI
//    //OPTIONAL: read back all device addresses
//    for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
//    {
//        ReadReg(currentBoard, DIR0_ADDR, response_frame2, 1, 0, FRMWRT_SGL_R);
//        printf("board %d\n",response_frame2[4]);
//    }

    //RESET ANY COMM FAULT CONDITIONS FROM STARTUP
    WriteReg(0, FAULT_RST2, 0x03, 1, FRMWRT_ALL_W);

    return;
}
//END AUTO ADDRESS SEQUENCE

//WRITE AND READ FUNCTIONS

//FORMAT WRITE DATA, SEND TO
//BE COMBINED WITH REST OF FRAME
int WriteReg(BYTE bID, uint16_t wAddr, uint64_t dwData, BYTE bLen, BYTE bWriteType) {
	// device address, register start address, data bytes, data length, write type (single, broadcast, stack)
	bRes = 0;
	memset(bBuf,0,sizeof(bBuf));
	switch (bLen) {
	case 1:
		bBuf[0] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 1, bWriteType);
		break;
	case 2:
		bBuf[0] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[1] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 2, bWriteType);
		break;
	case 3:
		bBuf[0] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[1] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[2] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 3, bWriteType);
		break;
	case 4:
		bBuf[0] = (dwData & 0x00000000FF000000) >> 24;
		bBuf[1] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[2] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[3] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 4, bWriteType);
		break;
	case 5:
		bBuf[0] = (dwData & 0x000000FF00000000) >> 32;
		bBuf[1] = (dwData & 0x00000000FF000000) >> 24;
		bBuf[2] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[3] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[4] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 5, bWriteType);
		break;
	case 6:
		bBuf[0] = (dwData & 0x0000FF0000000000) >> 40;
		bBuf[1] = (dwData & 0x000000FF00000000) >> 32;
		bBuf[2] = (dwData & 0x00000000FF000000) >> 24;
		bBuf[3] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[4] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[5] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 6, bWriteType);
		break;
	case 7:
		bBuf[0] = (dwData & 0x00FF000000000000) >> 48;
		bBuf[1] = (dwData & 0x0000FF0000000000) >> 40;
		bBuf[2] = (dwData & 0x000000FF00000000) >> 32;
		bBuf[3] = (dwData & 0x00000000FF000000) >> 24;
		bBuf[4] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[5] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[6] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 7, bWriteType);
		break;
	case 8:
		bBuf[0] = (dwData & 0xFF00000000000000) >> 56;
		bBuf[1] = (dwData & 0x00FF000000000000) >> 48;
		bBuf[2] = (dwData & 0x0000FF0000000000) >> 40;
		bBuf[3] = (dwData & 0x000000FF00000000) >> 32;
		bBuf[4] = (dwData & 0x00000000FF000000) >> 24;
		bBuf[5] = (dwData & 0x0000000000FF0000) >> 16;
		bBuf[6] = (dwData & 0x000000000000FF00) >> 8;
		bBuf[7] = dwData & 0x00000000000000FF;
		bRes = WriteFrame(bID, wAddr, bBuf, 8, bWriteType);
		break;
	default:
		break;
	}
	return bRes;
}

//GENERATE COMMAND FRAME
int WriteFrame(BYTE bID, uint16_t wAddr, BYTE * pData, BYTE bLen, BYTE bWriteType) {
	int bPktLen = 0;
	uint8_t * pBuf = pFrame;
	memset(pFrame, 0x7F, sizeof(pFrame));
	*pBuf++ = 0x80 | (bWriteType) | ((bWriteType & 0x10) ? bLen - 0x01 : 0x00); //Only include blen if it is a write; Writes are 0x90, 0xB0, 0xD0
	if (bWriteType == FRMWRT_SGL_R || bWriteType == FRMWRT_SGL_W)
	{
		*pBuf++ = (bID & 0x00FF);
	}
	*pBuf++ = (wAddr & 0xFF00) >> 8;
	*pBuf++ = wAddr & 0x00FF;

	while (bLen--)
		*pBuf++ = *pData++;

	bPktLen = pBuf - pFrame;

	wCRC = CRC16(pFrame, bPktLen);
	*pBuf++ = wCRC & 0x00FF;
	*pBuf++ = (wCRC & 0xFF00) >> 8;
	bPktLen += 2;
	//THIS SEEMS to occasionally drop bytes from the frame. Sometimes is not sending the last frame of the CRC.
	//(Seems to be caused by stack overflow, so take precautions to reduce stack usage in function calls)
	//sciSend(sciREG, bPktLen, pFrame);
	R_UART1_Send(pFrame,bPktLen);

	return bPktLen;
}

//GENERATE READ COMMAND FRAME AND THEN WAIT FOR RESPONSE DATA (INTERRUPT MODE FOR SCIRX)
int ReadReg(BYTE bID, uint16_t wAddr, BYTE * pData, BYTE bLen, uint32_t dwTimeOut,
        BYTE bWriteType) {
    // device address, register start address, byte frame pointer to store data, data length, read type (single, broadcast, stack)
    bRes = 0;
    count = 1000000; //timeout after this many attempts
    if (bWriteType == FRMWRT_SGL_R) {
        ReadFrameReq(bID, wAddr, bLen, bWriteType);
        memset(pData, 0, sizeof(pData));
        //sciEnableNotification(sciREG, SCI_RX_INT);//not need of this function because we permanently enable the uart receive interrupt
        //sciReceive(sciREG, bLen + 6, pData);
	R_UART1_Receive(pData, bLen + 6);
        while(UART_RX_RDY == 0U && count>0) count--; /* Wait */
        UART_RX_RDY = 0;
        bRes = bLen + 6;
    } else if (bWriteType == FRMWRT_STK_R) {
        bRes = ReadFrameReq(bID, wAddr, bLen, bWriteType);
        memset(pData, 0, sizeof(pData));
        //sciEnableNotification(sciREG, SCI_RX_INT);
        //sciReceive(sciREG, (bLen + 6) * (TOTALBOARDS - 1), pData);
	R_UART1_Receive(pData, (bLen + 6) * (TOTALBOARDS - 1));
        while(UART_RX_RDY == 0U && count>0) count--; /* Wait */
        UART_RX_RDY = 0;
        bRes = (bLen + 6) * (TOTALBOARDS - 1);
    } else if (bWriteType == FRMWRT_ALL_R) {
        bRes = ReadFrameReq(bID, wAddr, bLen, bWriteType);
        memset(pData, 0, sizeof(pData));
        //sciEnableNotification(sciREG, SCI_RX_INT);
        //sciReceive(sciREG, (bLen + 6) * TOTALBOARDS, pData);
	R_UART1_Receive(pData, (bLen + 6) * TOTALBOARDS);
        while(UART_RX_RDY == 0U && count>0) count--; /* Wait */
        UART_RX_RDY = 0;
        bRes = (bLen + 6) * TOTALBOARDS;
    } else {
        bRes = 0;
    }
// below commented in original file of TI
//    //CHECK IF CRC IS CORRECT
//    for(crc_i=0; crc_i<bRes; crc_i+=(bLen+6))
//    {
//        if(CRC16(&pData[crc_i], bLen+6)!=0)
//        {
//            printf("BAD CRC\n");
//        }
//    }

    return bRes;
}

int ReadFrameReq(BYTE bID, uint16_t wAddr, BYTE bByteToReturn, BYTE bWriteType) {
	bReturn = bByteToReturn - 1;

	if (bReturn > 127)
		return 0;

	return WriteFrame(bID, wAddr, &bReturn, 1, bWriteType);
}

// CRC16 TABLE
// ITU_T polynomial: x^16 + x^15 + x^2 + 1
const uint16_t crc16_table[256] = { 0x0000, 0xC0C1, 0xC181, 0x0140, 0xC301,
		0x03C0, 0x0280, 0xC241, 0xC601, 0x06C0, 0x0780, 0xC741, 0x0500, 0xC5C1,
		0xC481, 0x0440, 0xCC01, 0x0CC0, 0x0D80, 0xCD41, 0x0F00, 0xCFC1, 0xCE81,
		0x0E40, 0x0A00, 0xCAC1, 0xCB81, 0x0B40, 0xC901, 0x09C0, 0x0880, 0xC841,
		0xD801, 0x18C0, 0x1980, 0xD941, 0x1B00, 0xDBC1, 0xDA81, 0x1A40, 0x1E00,
		0xDEC1, 0xDF81, 0x1F40, 0xDD01, 0x1DC0, 0x1C80, 0xDC41, 0x1400, 0xD4C1,
		0xD581, 0x1540, 0xD701, 0x17C0, 0x1680, 0xD641, 0xD201, 0x12C0, 0x1380,
		0xD341, 0x1100, 0xD1C1, 0xD081, 0x1040, 0xF001, 0x30C0, 0x3180, 0xF141,
		0x3300, 0xF3C1, 0xF281, 0x3240, 0x3600, 0xF6C1, 0xF781, 0x3740, 0xF501,
		0x35C0, 0x3480, 0xF441, 0x3C00, 0xFCC1, 0xFD81, 0x3D40, 0xFF01, 0x3FC0,
		0x3E80, 0xFE41, 0xFA01, 0x3AC0, 0x3B80, 0xFB41, 0x3900, 0xF9C1, 0xF881,
		0x3840, 0x2800, 0xE8C1, 0xE981, 0x2940, 0xEB01, 0x2BC0, 0x2A80, 0xEA41,
		0xEE01, 0x2EC0, 0x2F80, 0xEF41, 0x2D00, 0xEDC1, 0xEC81, 0x2C40, 0xE401,
		0x24C0, 0x2580, 0xE541, 0x2700, 0xE7C1, 0xE681, 0x2640, 0x2200, 0xE2C1,
		0xE381, 0x2340, 0xE101, 0x21C0, 0x2080, 0xE041, 0xA001, 0x60C0, 0x6180,
		0xA141, 0x6300, 0xA3C1, 0xA281, 0x6240, 0x6600, 0xA6C1, 0xA781, 0x6740,
		0xA501, 0x65C0, 0x6480, 0xA441, 0x6C00, 0xACC1, 0xAD81, 0x6D40, 0xAF01,
		0x6FC0, 0x6E80, 0xAE41, 0xAA01, 0x6AC0, 0x6B80, 0xAB41, 0x6900, 0xA9C1,
		0xA881, 0x6840, 0x7800, 0xB8C1, 0xB981, 0x7940, 0xBB01, 0x7BC0, 0x7A80,
		0xBA41, 0xBE01, 0x7EC0, 0x7F80, 0xBF41, 0x7D00, 0xBDC1, 0xBC81, 0x7C40,
		0xB401, 0x74C0, 0x7580, 0xB541, 0x7700, 0xB7C1, 0xB681, 0x7640, 0x7200,
		0xB2C1, 0xB381, 0x7340, 0xB101, 0x71C0, 0x7080, 0xB041, 0x5000, 0x90C1,
		0x9181, 0x5140, 0x9301, 0x53C0, 0x5280, 0x9241, 0x9601, 0x56C0, 0x5780,
		0x9741, 0x5500, 0x95C1, 0x9481, 0x5440, 0x9C01, 0x5CC0, 0x5D80, 0x9D41,
		0x5F00, 0x9FC1, 0x9E81, 0x5E40, 0x5A00, 0x9AC1, 0x9B81, 0x5B40, 0x9901,
		0x59C0, 0x5880, 0x9841, 0x8801, 0x48C0, 0x4980, 0x8941, 0x4B00, 0x8BC1,
		0x8A81, 0x4A40, 0x4E00, 0x8EC1, 0x8F81, 0x4F40, 0x8D01, 0x4DC0, 0x4C80,
		0x8C41, 0x4400, 0x84C1, 0x8581, 0x4540, 0x8701, 0x47C0, 0x4680, 0x8641,
		0x8201, 0x42C0, 0x4380, 0x8341, 0x4100, 0x81C1, 0x8081, 0x4040 };

uint16_t CRC16(BYTE *pBuf, int nLen) {
    uint16_t wCRC = 0xFFFF;
    int i;

    for (i = 0; i < nLen; i++) {
        wCRC ^= (*pBuf++) & 0x00FF;
        wCRC = crc16_table[wCRC & 0x00FF] ^ (wCRC >> 8);
    }

    return wCRC;
}

////END WRITE AND READ FUNCTIONS

////MISCELLANEOUS FUNCTIONS

//BOOL GetFaultStat() {

//    if (!gioGetBit(gioPORTA, 0))
//        return 0;
//    return 1;
//}

//void delayus(uint16 us) {
//    if (us == 0)
//       return;
//    else
//    {
//        //CHANGE THE INTERRUPT COMPARE VALUES (PERIOD OF INTERRUPT)
//        //Setup compare 0 value.
//        rtiREG1->CMP[0U].COMPx = 10*us; //10 ticks of clock per microsecond, so multiply by 10
//        //Setup update compare 0 value.
//        rtiREG1->CMP[0U].UDCPx = 10*us;

//        //ENABLE THE NOTIFICATION FOR THE PERIOD WE SET
//        rtiEnableNotification(rtiNOTIFICATION_COMPARE0);

//        //START THE COUNTER
//        rtiStartCounter(rtiCOUNTER_BLOCK0);

//        //WAIT IN LOOP UNTIL THE INTERRUPT HAPPENS (HAPPENS AFTER THE PERIOD WE SET)
//        //WHEN INTERRUPT HAPPENS, RTI_NOTIFICATION GETS SET TO 1 IN THAT INTERRUPT
//        //GO TO notification.c -> rtiNotification() to see where RTI_TIMEOUT is set to 1
//        while(RTI_TIMEOUT==0);

//        //RESET THE VARIABLE TO 0, FOR THE NEXT TIME WE DO A DELAY
//        RTI_TIMEOUT = 0;

//        //DISABLE THE INTERRUPT NOTIFICATION
//        rtiDisableNotification(rtiNOTIFICATION_COMPARE0);

//        //STOP THE COUNTER
//        rtiStopCounter(rtiCOUNTER_BLOCK0);

//        //RESET COUNTER FOR THE NEXT TIME WE DO A DELAY
//        rtiResetCounter(rtiCOUNTER_BLOCK0);
//    }
//}

//void delayms(uint16 ms) {
//    if (ms == 0)
//       return;
//    else
//    {
//        rtiREG1->CMP[0U].COMPx = 10000*ms;
//        rtiREG1->CMP[0U].UDCPx = 10000*ms;
//        rtiEnableNotification(rtiNOTIFICATION_COMPARE0);
//        rtiStartCounter(rtiCOUNTER_BLOCK0);
//        while(RTI_TIMEOUT==0);
//        RTI_TIMEOUT = 0;
//        rtiDisableNotification(rtiNOTIFICATION_COMPARE0);
//        rtiStopCounter(rtiCOUNTER_BLOCK0);
//        rtiResetCounter(rtiCOUNTER_BLOCK0);
//    }
//}

void ResetAllFaults(BYTE bID, BYTE bWriteType)
{
    //BROADCAST INCLUDES EXTRA FUNCTIONALITY TO OVERWRITE THE CUST_CRC WITH THE CURRENT SETTINGS
    if(bWriteType==FRMWRT_ALL_W)
    {
        //READ THE CALCULATED CUSTOMER CRC VALUES
        ReadReg(0, CUST_CRC_RSLT_HI, fault_frame, 2, 0, FRMWRT_ALL_R);
        //OVERWRITE THE CRC OF EVERY BOARD IN THE STACK WITH THE CORRECT CRC
        for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
        {
            //THE RETURN FRAME STARTS WITH THE HIGHEST BOARD FIRST, SO THIS WILL WRITE THE HIGHEST BOARD FIRST
            WriteReg(TOTALBOARDS-currentBoard-1, CUST_CRC_HI, fault_frame[currentBoard*8+4] << 8 | fault_frame[currentBoard*8+5], 2, FRMWRT_SGL_W);
        }
        //NOW CLEAR EVERY FAULT
        WriteReg(0, FAULT_RST1, 0xFFFF, 2, FRMWRT_ALL_W);
    }
    else if(bWriteType==FRMWRT_SGL_W)
    {
        WriteReg(bID, FAULT_RST1, 0xFFFF, 2, FRMWRT_SGL_W);
    }
    else if(bWriteType==FRMWRT_STK_W)
    {
        WriteReg(0, FAULT_RST1, 0xFFFF, 2, FRMWRT_STK_W);
    }
    else
    {
        printf("ERROR: ResetAllFaults bWriteType incorrect\n");
    }
}

//void MaskAllFaults(BYTE bID, BYTE bWriteType)
//{
//    if(bWriteType==FRMWRT_ALL_W)
//    {
//        WriteReg(0, FAULT_MSK1, 0xFFFF, 2, FRMWRT_ALL_W);
//    }
//    else if(bWriteType==FRMWRT_SGL_W)
//    {
//        WriteReg(bID, FAULT_MSK1, 0xFFFF, 2, FRMWRT_SGL_W);
//    }
//    else if(bWriteType==FRMWRT_STK_W)
//    {
//        WriteReg(0, FAULT_MSK1, 0xFFFF, 2, FRMWRT_STK_W);
//    }
//    else
//    {
//        printf("ERROR: MaskAllFaults bWriteType incorrect\n");
//    }
//}

//void PrintAllFaults(BYTE bID, BYTE bWriteType)
//{
//    //PRINT 39 REGISTERS STARTING FROM FAULT_SUMMARY (INCLUDES RESERVED REGISTERS)
//    printf("\n");
//    currentBoard = 0;
//    currentCell = 0;
//    memset(fault_frame,0,sizeof(fault_frame));
//    if(bWriteType==FRMWRT_ALL_R)
//    {
//        ReadReg(0, FAULT_SUMMARY, fault_frame, 39, 0, FRMWRT_ALL_R);
//        for(currentBoard = 0; currentBoard<TOTALBOARDS; currentBoard++)
//        {
//            printf("BOARD %d FAULTS:\t",TOTALBOARDS-currentBoard);
//            for(currentCell = 0; currentCell<39; currentCell++)
//            {
//                printf("%02x ",fault_frame[(currentBoard*(39+6))+4+currentCell]);
//            }
//            printf("\n");
//        }
//    }
//    else if(bWriteType==FRMWRT_SGL_R)
//    {
//        ReadReg(bID, FAULT_SUMMARY, fault_frame, 39, 0, FRMWRT_SGL_R);
//        printf("BOARD %d FAULTS:\t",bID);
//        for(currentCell = 0; currentCell<39; currentCell++)
//        {
//            printf("%02x ",fault_frame[4+currentCell]);
//        }
//        printf("\n");
//    }
//    else if(bWriteType==FRMWRT_STK_R)
//    {
//        ReadReg(0, FAULT_SUMMARY, fault_frame, 39, 0, FRMWRT_STK_R);
//        for(currentBoard = 0; currentBoard<(TOTALBOARDS-1); currentBoard++)
//        {
//            printf("BOARD %d FAULTS:\t",TOTALBOARDS-currentBoard);
//            for(currentCell = 0; currentCell<39; currentCell++)
//            {
//                printf("%02x ",fault_frame[(currentBoard*(39+6))+4+currentCell]);
//            }
//            printf("\n");
//        }
//    }
//    else
//    {
//        printf("ERROR: PrintAllFaults bWriteType incorrect\n");
//    }
//    printf("\n");
//}

////RUN BASIC CELL BALANCING FOR ALL DEVICES
//void RunCB()
//{
//    //SET BALANCING TIMERS TO 30 s
//    WriteReg(0, CB_CELL16_CTRL, 0x0202020202020202, 8, FRMWRT_ALL_W);   //cell 16-9 (8 byte max write)
//    WriteReg(0, CB_CELL8_CTRL, 0x0202020202020202, 8, FRMWRT_ALL_W);    //cell 8-1

//    //SET DUTY CYCLE TO 10 s (default)
//    WriteReg(0, BAL_CTRL1, 0x01, 1, FRMWRT_ALL_W);   //10s duty cycle

//    //OPTIONAL: SET VCBDONE THRESH TO 3V, AND OVUV_GO
//    WriteReg(0, VCB_DONE_THRESH, 0x08, 1, FRMWRT_ALL_W);    //3V threshold (8*25mV + 2.8V)
//    WriteReg(0, OVUV_CTRL, 0x05, 1, FRMWRT_ALL_W);          //round-robin and OVUV_GO

//    //START BALANCING
//    WriteReg(0, BAL_CTRL2, 0x03, 1, FRMWRT_ALL_W); //auto balance and BAL_GO
//}

////RUN BASIC REVERSE ADDRESSING SEQUENCE
//void ReverseAddressing()
//{
//    //CHANGE BASE DEVICE DIRECTION
//    WriteReg(0, CONTROL1, 0x80, 1, FRMWRT_SGL_W);

//    //CHANGE REST OF STACK DIRECTION
//    WriteReg(0, CONTROL1, 0x80, 1, FRMWRT_REV_ALL_W);

//    //DO NORMAL AUTO ADDRESS SEQUENCE, BUT FOR DIR1_ADDR
//    //DUMMY WRITE TO SNCHRONIZE ALL DAISY CHAIN DEVICES DLL (IF A DEVICE RESET OCCURED PRIOR TO THIS)
//    WriteReg(0, OTP_ECC_DATAIN1, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN2, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN3, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN4, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN5, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN6, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN7, 0X00, 1, FRMWRT_ALL_W);
//    WriteReg(0, OTP_ECC_DATAIN8, 0X00, 1, FRMWRT_ALL_W);

//    //ENABLE AUTO ADDRESSING MODE, WHILE KEEPING REVERSE DIRECTION
//    WriteReg(0, CONTROL1, 0X81, 1, FRMWRT_ALL_W);

//    //SET ADDRESSES FOR EVERY BOARD (REVERSE DIRECTION)
//    for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
//    {
//        WriteReg(0, DIR1_ADDR, currentBoard, 1, FRMWRT_ALL_W);
//    }

//    WriteReg(0, COMM_CTRL, 0x02, 1, FRMWRT_ALL_W); //set everything as a stack device first

//    if(TOTALBOARDS==1) //if there's only 1 board, it's the base AND top of stack, so change it to those
//    {
//        WriteReg(0, COMM_CTRL, 0x01, 1, FRMWRT_SGL_W);
//    }
//    else //otherwise set the base and top of stack individually
//    {
//        WriteReg(0, COMM_CTRL, 0x00, 1, FRMWRT_SGL_W);
//        WriteReg(TOTALBOARDS-1, COMM_CTRL, 0x03, 1, FRMWRT_SGL_W);
//    }

//    //SYNCRHONIZE THE DLL WITH A THROW-AWAY READ
//    ReadReg(0, OTP_ECC_DATAIN1, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN2, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN3, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN4, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN5, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN6, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN7, response_frame2, 1, 0, FRMWRT_ALL_R);
//    ReadReg(0, OTP_ECC_DATAIN8, response_frame2, 1, 0, FRMWRT_ALL_R);

//    ////OPTIONAL: read back all device addresses
//    //for(currentBoard=0; currentBoard<TOTALBOARDS; currentBoard++)
//    //{
//    //    ReadReg(currentBoard, DIR0_ADDR, response_frame2, 1, 0, FRMWRT_SGL_R);
//    //    printf("board %d\n",response_frame2[4]);
//    //}

//    //RESET ANY COMM FAULT CONDITIONS FROM STARTUP
//    WriteReg(0, FAULT_RST2, 0x03, 1, FRMWRT_ALL_W);
//}
////END MISCELLANEOUS FUNCTIONS

////EOF

您好、TI 团队、

目前、我正在将 BQ79656-Q1 IC 用于汽车 BMS 应用。 我们在 BAT 引脚提供56伏电压。 通过使用 ping 命令、可得到 LDOIN=6.0V、AVDD=5V、REFHP=5V、TSREF=0V、NEG5V=-4.7V、CVDD=5V、DVDD=1.8V。 在这个阶段,一切都很好。

我将1个 BQ79656-Q1 BMS IC 与 R5F10BGG 主机控制器配合使用。 我正在使用波特率为1Mbps 的 UART 通信。 将 BQ79616-Q1的"TMS570"参考代码移植到 R5F10BGG 微控制器中。

现在、问题是我无法读取电池电压。 我正在发送数据"0xC0、0x05、0x68、0x1F、0x42、0x2D"以读取电芯电压。 我 还会检查从主机控制器传输到 CRO 上的 BQ79656-Q1以及 USBTOTTL 转换器的数据。 此外、还会在 BQ BMS IC 的 RX 引脚接收数据。   

请帮助我解决此问题。

在此代码中、我们将使用自动寻址功能。 还有 一点我想知道、我们如何知道我们的器件地址(BQ79656-Q1)?

在传输的字符串 "0xC0、0x05、0x68、0x1F、0x42、0x2D"中、我认为地址是5。

我也在分享代码的某些部分……  

此致、谢谢。

迪帕克

  • 请注意,本文内容源自机器翻译,可能存在语法或其它翻译错误,仅供参考。如需获取准确内容,请参阅链接中的英语原文或自行翻译。

    Deepak,

    您可以从我们的 数据表的第9.3.6.1.1.2节了解有关我们的通信帧的更多信息。 您共享的命令是广播读取、因此不包含器件地址。 0x05是寄存器地址0x568的第一部分、它是第一个存储电芯电压的寄存器 VCELL16_HI 的地址。 特定器件的地址分别可在寄存器0x306 (DIR0_ADDR)或0x307 (DIR1_ADDR)中找到、用于正向或反向通信方向。 但是、在被称为自动寻址的过程中为器件指定地址后、才能读取寄存器。 数据表的第9.3.6.1.3节详细介绍了启动过程、但简而言之、第一个器件菊花链将是器件0、第二个器件1、依此类推。  

    您的问题很可能是由以下两个原因之一导致的:

    1) 1)您的 MCU 和 BQ79656之间存在通信故障。 这很可能是硬件设置的问题。 如果您使用的是 EVM、则在器件接收到唤醒 ping 时、芯片附近会有一个绿灯亮起。 如果此 LED 不亮起、则可能是问题的原因。 您应该向我发送您的硬件设置的图片、以便我帮助您修复它。

    2)自动寻址功能失败。 如果设备正在唤醒、但无法与之通信、则问题通常出在此处。 从 bq79616.c 的第171行开始打开可选器件读回、并使用逻辑分析仪探测 BQ79656的 RX 和 TX 引脚、是诊断该问题的最简单方法。

    此致、