mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-25 09:07:23 +08:00
470 lines
15 KiB
C
470 lines
15 KiB
C
//###########################################################################
|
|
//
|
|
// FILE: F2837xD_Ipc_Driver_Util.c
|
|
//
|
|
// TITLE: F2837xD Inter-Processor Communication (IPC) API Driver Utility
|
|
// Functions
|
|
//
|
|
// DESCRIPTION:
|
|
// API functions for inter-processor communications between the
|
|
// Local and Remote CPU system.
|
|
// The driver functions in this file are available only as
|
|
// sample functions for application development. Due to the generic
|
|
// nature of these functions and the cycle overhead inherent to a
|
|
// function call, the code is not intended to be used in cases where
|
|
// maximum efficiency is required in a system.
|
|
//
|
|
// NOTE: This source code is used by both CPUs. That is both CPU1 and CPU2
|
|
// cores use this code.
|
|
// The active debug CPU will be referred to as Local CPU and the other
|
|
// CPU will be referred to as Remote CPU.
|
|
// When using this source code in CPU1, the term "local"
|
|
// will mean CPU1 and the term "remote" CPU will be mean CPU2.
|
|
// When using this source code in CPU2, the term "local"
|
|
// will mean CPU2 and the term "remote" CPU will be mean CPU1.
|
|
//
|
|
// The abbreviations LtoR and RtoL within the function names mean
|
|
// Local to Remote and Remote to Local respectively.
|
|
//
|
|
//###########################################################################
|
|
// $TI Release: F2837xD Support Library v3.05.00.00 $
|
|
// $Release Date: Tue Jun 26 03:15:23 CDT 2018 $
|
|
// $Copyright:
|
|
// Copyright (C) 2013-2018 Texas Instruments Incorporated - http://www.ti.com/
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions
|
|
// are met:
|
|
//
|
|
// Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
//
|
|
// Redistributions in binary form must reproduce the above copyright
|
|
// notice, this list of conditions and the following disclaimer in the
|
|
// documentation and/or other materials provided with the
|
|
// distribution.
|
|
//
|
|
// Neither the name of Texas Instruments Incorporated nor the names of
|
|
// its contributors may be used to endorse or promote products derived
|
|
// from this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
// $
|
|
//###########################################################################
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! \addtogroup ipc_util_api
|
|
//! @{
|
|
//
|
|
//*****************************************************************************
|
|
#include "F2837xD_device.h"
|
|
#include "F2837xD_GlobalPrototypes.h"
|
|
#include "F2837xD_Gpio_defines.h"
|
|
#include "F2837xD_Ipc_drivers.h"
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Local CPU Acknowledges Remote to Local IPC Flag.
|
|
//!
|
|
//! \param ulFlags specifies the IPC flag mask for flags being acknowledged.
|
|
//!
|
|
//! This function will allow the Local CPU system to acknowledge/clear the IPC
|
|
//! flag set by the Remote CPU system. The \e ulFlags parameter can be any of
|
|
//! the IPC flag values: \b IPC_FLAG0 - \b IPC_FLAG31.
|
|
//!
|
|
//! \return None.
|
|
//
|
|
//*****************************************************************************
|
|
void
|
|
IPCRtoLFlagAcknowledge (uint32_t ulFlags)
|
|
{
|
|
IpcRegs.IPCACK.all |= ulFlags;
|
|
}
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Determines whether the given Remote to Local IPC flags are busy or not.
|
|
//!
|
|
//! \param ulFlags specifies Remote to Local IPC Flag number masks to check the
|
|
//! status of.
|
|
//!
|
|
//! Allows the caller to determine whether the designated IPC flags are
|
|
//! pending. The \e ulFlags parameter can be any of the IPC flag
|
|
//! values: \b IPC_FLAG0 - \b IPC_FLAG31.
|
|
//!
|
|
//! \return Returns \b 1 if the IPC flags are busy or \b 0 if designated
|
|
//! IPC flags are free.
|
|
//
|
|
//*****************************************************************************
|
|
Uint16
|
|
IPCRtoLFlagBusy (uint32_t ulFlags)
|
|
{
|
|
Uint16 returnStatus;
|
|
|
|
if ((IpcRegs.IPCSTS.all & ulFlags) == 0)
|
|
{
|
|
returnStatus = 0;
|
|
}
|
|
else
|
|
{
|
|
returnStatus = 1;
|
|
}
|
|
|
|
return returnStatus;
|
|
}
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Determines whether the given IPC flags are busy or not.
|
|
//!
|
|
//! \param ulFlags specifies Local to Remote IPC Flag number masks to check the
|
|
//! status of.
|
|
//!
|
|
//! Allows the caller to determine whether the designated IPC flags are
|
|
//! available for further control to master system communication. If \b 0 is
|
|
//! returned, then all designated tasks have completed and are available.
|
|
//! The \e ulFlags parameter can be any of the IPC flag
|
|
//! values: \b IPC_FLAG0 - \b IPC_FLAG31.
|
|
//!
|
|
//! \return Returns \b 1 if the IPC flags are busy or \b 0 if designated
|
|
//! IPC flags are free.
|
|
//
|
|
//*****************************************************************************
|
|
Uint16
|
|
IPCLtoRFlagBusy (uint32_t ulFlags)
|
|
{
|
|
Uint16 returnStatus;
|
|
|
|
if ((IpcRegs.IPCFLG.all & ulFlags) == 0)
|
|
{
|
|
returnStatus = 0;
|
|
}
|
|
else
|
|
{
|
|
returnStatus = 1;
|
|
}
|
|
|
|
return returnStatus;
|
|
}
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Local CPU Sets Local to Remote IPC Flag
|
|
//!
|
|
//! \param ulFlags specifies the IPC flag mask for flags being set.
|
|
//!
|
|
//! This function will allow the Local CPU system to set the designated IPC
|
|
//! flags to send to the Remote CPU system. The \e ulFlags parameter can be any
|
|
//! of the IPC flag values: \b IPC_FLAG0 - \b IPC_FLAG31.
|
|
//!
|
|
//! \return None.
|
|
//
|
|
//*****************************************************************************
|
|
void
|
|
IPCLtoRFlagSet (uint32_t ulFlags)
|
|
{
|
|
IpcRegs.IPCSET.all |= ulFlags;
|
|
}
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Local CPU Clears Local to Remote IPC Flag
|
|
//!
|
|
//! \param ulFlags specifies the IPC flag mask for flags being set.
|
|
//!
|
|
//! This function will allow the Local CPU system to set the designated IPC
|
|
//! flags to send to the Remote CPU system. The \e ulFlags parameter can be any
|
|
//! of the IPC flag values: \b IPC_FLAG0 - \b IPC_FLAG31.
|
|
//!
|
|
//! \return None.
|
|
//
|
|
//*****************************************************************************
|
|
void
|
|
IPCLtoRFlagClear (uint32_t ulFlags)
|
|
{
|
|
IpcRegs.IPCCLR.all |= ulFlags;
|
|
}
|
|
|
|
//*****************************************************************************
|
|
//
|
|
//! Local Return CPU02 BOOT status
|
|
//!
|
|
//! This function returns the value at IPCBOOTSTS register.
|
|
//!
|
|
//! \return Boot status.
|
|
//
|
|
//*****************************************************************************
|
|
uint32_t
|
|
IPCGetBootStatus (void)
|
|
{
|
|
return(IpcRegs.IPCBOOTSTS);
|
|
}
|
|
|
|
#if defined (CPU1)
|
|
//*****************************************************************************
|
|
//! Executes a CPU02 control system bootloader.
|
|
//!
|
|
//! \param ulBootMode specifies which CPU02 control system boot mode to execute.
|
|
//!
|
|
//! This function will allow the CPU01 master system to boot the CPU02 control
|
|
//! system via the following modes: Boot to RAM, Boot to Flash, Boot via SPI,
|
|
//! SCI, I2C, or parallel I/O. Unlike other IPCLite driver functions, this
|
|
//! function blocks and waits until the control system boot ROM is configured
|
|
//! and ready to receive CPU01 to CPU02 IPC INT0 interrupts. It then blocks and
|
|
//! waits until IPC INT0 and IPC FLAG31 are available in the CPU02 boot ROM
|
|
//! prior to sending the command to execute the selected bootloader. The \e
|
|
//! ulBootMode parameter accepts one of the following values: \b
|
|
//! C1C2_BROM_BOOTMODE_BOOT_FROM_PARALLEL, \b
|
|
//! C1C2_BROM_BOOTMODE_BOOT_FROM_SCI, \b
|
|
//! C1C2_BROM_BOOTMODE_BOOT_FROM_SPI, \b
|
|
//! C1C2_BROM_BOOTMODE_BOOT_FROM_I2C, \b C1C2_BROM_BOOTMODE_BOOT_FROM_CAN,
|
|
//! \b C1C2_BROM_BOOTMODE_BOOT_FROM_RAM, \b
|
|
//! C1C2_BROM_BOOTMODE_BOOT_FROM_FLASH.
|
|
//!
|
|
//! \return 0 (success) if command is sent, or 1 (failure) if boot mode is
|
|
//! invalid and command was not sent.
|
|
//
|
|
//*****************************************************************************
|
|
uint16_t
|
|
IPCBootCPU2(uint32_t ulBootMode)
|
|
{
|
|
uint32_t bootStatus;
|
|
uint16_t pin;
|
|
uint16_t returnStatus = STATUS_PASS;
|
|
|
|
//
|
|
// If CPU2 has already booted, return a fail to let the application
|
|
// know that something is out of the ordinary.
|
|
//
|
|
bootStatus = IPCGetBootStatus() & 0x0000000F;
|
|
|
|
if(bootStatus == C2_BOOTROM_BOOTSTS_C2TOC1_BOOT_CMD_ACK)
|
|
{
|
|
//
|
|
// Check if MSB is set as well
|
|
//
|
|
bootStatus = ((uint32_t)(IPCGetBootStatus() & 0x80000000)) >> 31U;
|
|
|
|
if(bootStatus != 0)
|
|
{
|
|
returnStatus = STATUS_FAIL;
|
|
|
|
return returnStatus;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Wait until CPU02 control system boot ROM is ready to receive
|
|
// CPU01 to CPU02 INT1 interrupts.
|
|
//
|
|
do
|
|
{
|
|
bootStatus = IPCGetBootStatus() & C2_BOOTROM_BOOTSTS_SYSTEM_READY;
|
|
} while ((bootStatus != C2_BOOTROM_BOOTSTS_SYSTEM_READY));
|
|
|
|
//
|
|
// Loop until CPU02 control system IPC flags 1 and 32 are available
|
|
//
|
|
while ((IPCLtoRFlagBusy(IPC_FLAG0) == 1) ||
|
|
(IPCLtoRFlagBusy(IPC_FLAG31) == 1))
|
|
{
|
|
|
|
}
|
|
|
|
if (ulBootMode >= C1C2_BROM_BOOTMODE_BOOT_COMMAND_MAX_SUPPORT_VALUE)
|
|
{
|
|
returnStatus = STATUS_FAIL;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Based on boot mode, enable pull-ups on peripheral pins and
|
|
// give GPIO pin control to CPU02 control system.
|
|
//
|
|
switch (ulBootMode)
|
|
{
|
|
case C1C2_BROM_BOOTMODE_BOOT_FROM_SCI:
|
|
|
|
EALLOW;
|
|
|
|
//
|
|
//SCIA connected to CPU02
|
|
//
|
|
DevCfgRegs.CPUSEL5.bit.SCI_A = 1;
|
|
|
|
//
|
|
//Allows CPU02 bootrom to take control of clock
|
|
//configuration registers
|
|
//
|
|
ClkCfgRegs.CLKSEM.all = 0xA5A50000;
|
|
|
|
ClkCfgRegs.LOSPCP.all = 0x0002;
|
|
EDIS;
|
|
|
|
GPIO_SetupPinOptions(29, GPIO_OUTPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(29,GPIO_MUX_CPU2,1);
|
|
|
|
GPIO_SetupPinOptions(28, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(28,GPIO_MUX_CPU2,1);
|
|
|
|
break;
|
|
|
|
case C1C2_BROM_BOOTMODE_BOOT_FROM_SPI:
|
|
EALLOW;
|
|
|
|
//
|
|
//SPI-A connected to CPU02
|
|
//
|
|
DevCfgRegs.CPUSEL6.bit.SPI_A = 1;
|
|
|
|
//
|
|
//Allows CPU02 bootrom to take control of clock configuration
|
|
// registers
|
|
//
|
|
ClkCfgRegs.CLKSEM.all = 0xA5A50000;
|
|
EDIS;
|
|
|
|
GPIO_SetupPinOptions(16, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(16,GPIO_MUX_CPU2,1);
|
|
|
|
GPIO_SetupPinOptions(17, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(17,GPIO_MUX_CPU2,1);
|
|
|
|
GPIO_SetupPinOptions(18, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(18,GPIO_MUX_CPU2,1);
|
|
|
|
GPIO_SetupPinOptions(19, GPIO_OUTPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(19,GPIO_MUX_CPU2,0);
|
|
|
|
break;
|
|
|
|
case C1C2_BROM_BOOTMODE_BOOT_FROM_I2C:
|
|
EALLOW;
|
|
|
|
//
|
|
//I2CA connected to CPU02
|
|
//
|
|
DevCfgRegs.CPUSEL7.bit.I2C_A = 1;
|
|
|
|
//
|
|
//Allows CPU2 bootrom to take control of clock
|
|
//configuration registers
|
|
//
|
|
ClkCfgRegs.CLKSEM.all = 0xA5A50000;
|
|
ClkCfgRegs.LOSPCP.all = 0x0002;
|
|
EDIS;
|
|
GPIO_SetupPinOptions(32, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(32,GPIO_MUX_CPU2,1);
|
|
|
|
GPIO_SetupPinOptions(33, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(33,GPIO_MUX_CPU2,1);
|
|
|
|
break;
|
|
case C1C2_BROM_BOOTMODE_BOOT_FROM_PARALLEL:
|
|
|
|
for(pin=58;pin<=65;pin++)
|
|
{
|
|
GPIO_SetupPinOptions(pin, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(pin,GPIO_MUX_CPU2,0);
|
|
}
|
|
|
|
GPIO_SetupPinOptions(69, GPIO_OUTPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(69,GPIO_MUX_CPU2,0);
|
|
|
|
GPIO_SetupPinOptions(70, GPIO_INPUT, GPIO_ASYNC);
|
|
GPIO_SetupPinMux(70,GPIO_MUX_CPU2,0);
|
|
break;
|
|
|
|
|
|
case C1C2_BROM_BOOTMODE_BOOT_FROM_CAN:
|
|
//
|
|
//Set up the GPIO mux to bring out CANATX on GPIO71
|
|
//and CANARX on GPIO70
|
|
//
|
|
EALLOW;
|
|
GpioCtrlRegs.GPCLOCK.all = 0x00000000; //Unlock GPIOs 64-95
|
|
|
|
//
|
|
//Give CPU2 control just in case
|
|
//
|
|
GpioCtrlRegs.GPCCSEL1.bit.GPIO71 = GPIO_MUX_CPU2;
|
|
|
|
//
|
|
//Set the extended mux to 0x5
|
|
//
|
|
GpioCtrlRegs.GPCGMUX1.bit.GPIO71 = 0x1;
|
|
GpioCtrlRegs.GPCMUX1.bit.GPIO71 = 0x1;
|
|
|
|
//
|
|
//Set qualification to async just in case
|
|
//
|
|
GpioCtrlRegs.GPCQSEL1.bit.GPIO71 = 0x3;
|
|
|
|
GpioCtrlRegs.GPCLOCK.all = 0x00000000; //Unlock GPIOs 64-95
|
|
|
|
//
|
|
//Give CPU2 control just in case
|
|
//
|
|
GpioCtrlRegs.GPCCSEL1.bit.GPIO70 = GPIO_MUX_CPU2;
|
|
|
|
//
|
|
//Set the extended mux to bring out CANATX
|
|
//
|
|
GpioCtrlRegs.GPCGMUX1.bit.GPIO70 = 0x1;
|
|
GpioCtrlRegs.GPCMUX1.bit.GPIO70 = 0x1;
|
|
|
|
//
|
|
//Set qualification to async just in case
|
|
//
|
|
GpioCtrlRegs.GPCQSEL1.bit.GPIO70 = 0x3;
|
|
GpioCtrlRegs.GPCLOCK.all = 0xFFFFFFFF; //Lock GPIOs 64-95
|
|
ClkCfgRegs.CLKSRCCTL2.bit.CANABCLKSEL = 0x0;
|
|
CpuSysRegs.PCLKCR10.bit.CAN_A = 1;
|
|
EDIS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
//
|
|
//CPU01 to CPU02 IPC Boot Mode Register
|
|
//
|
|
IpcRegs.IPCBOOTMODE = ulBootMode;
|
|
|
|
//
|
|
// CPU01 To CPU02 IPC Command Register
|
|
//
|
|
IpcRegs.IPCSENDCOM = BROM_IPC_EXECUTE_BOOTMODE_CMD;
|
|
|
|
//
|
|
// CPU01 to CPU02 IPC flag register
|
|
//
|
|
IpcRegs.IPCSET.all = 0x80000001;
|
|
|
|
}
|
|
|
|
|
|
|
|
return returnStatus;
|
|
}
|
|
|
|
|
|
#endif
|
|
//*****************************************************************************
|
|
// Close the Doxygen group.
|
|
//! @}
|
|
//*****************************************************************************
|
|
|
|
|