工程师们好,最近我将Mcbsp0配置为spi从设备,相关寄存器配置如下所示(参考 SPRU592E,第6.7节)。代码的注释并不准确,DSP在时钟下降沿接收数据:
MCBSP_Config SPI_SLAVE = {
MCBSP_SPCR1_RMK(
MCBSP_SPCR1_DLB_OFF, /* DLB = 0,禁止自闭环方式 */
MCBSP_SPCR1_RJUST_LZF, /* RJUST = 2,左检验,低位填0*/
MCBSP_SPCR1_CLKSTP_NODELAY, /时停模式无延迟*/
MCBSP_SPCR1_DXENA_NA, /* DXENA = 1,不允许DX数据延迟 */
0, /* ABIS = 0 */
MCBSP_SPCR1_RINTM_FRM, /* RINTM = 0 ,RRDY bit从0到1产生接收中断*/
0, /* RSYNCER = 0,无同步错误*/
MCBSP_SPCR1_RRST_DISABLE /* RRST = 0,串口复位 */
),
MCBSP_SPCR2_RMK(
MCBSP_SPCR2_FREE_NO, /* FREE = 0,时钟受SOFYT bit控制 */
MCBSP_SPCR2_SOFT_NO, /* SOFT = 0,调试出现断点时时钟立即停止 */
//MCBSP_SPCR2_FRST_RESET, //帧同步关闭
MCBSP_SPCR2_FRST_FSG, /* FRST = 0 */
//MCBSP_SPCR2_GRST_RESET, /* GRST = 0,采样率发生器复位,CLKG频率为CPU时钟周期的一半 */
MCBSP_SPCR2_GRST_CLKG,
MCBSP_SPCR2_XINTM_XRDY, /* XINTM = 0,XRDY bit从0到1产生发送中断 */
0, /* XSYNCER = N/A,无同步错误 */
MCBSP_SPCR2_XRST_DISABLE /* XRST = 0,发送串口复位 */
),
/*单数据相,接受数据长度为16位,每相2个数据*/
MCBSP_RCR1_RMK(
MCBSP_RCR1_RFRLEN1_OF(0), /* RFRLEN1 = 0,接收帧每帧一字 */
MCBSP_RCR1_RWDLEN1_16BIT /* RWDLEN1 = 5,每字32位 */
),
MCBSP_RCR2_RMK(
MCBSP_RCR2_RPHASE_SINGLE, /* RPHASE = 0,单段帧 */
MCBSP_RCR2_RFRLEN2_OF(0), /* RFRLEN2 = 0,该帧无字 */
MCBSP_RCR2_RWDLEN2_8BIT, /* RWDLEN2 = 0,每字8bit */
MCBSP_RCR2_RCOMPAND_MSB, /* RCOMPAND = 0,无压缩,先传输高位数据 */
MCBSP_RCR2_RFIG_YES, /* RFIG = 0,每个帧同步信号启动一次数据接收 */
MCBSP_RCR2_RDATDLY_0BIT /* RDATDLY = 1,1bit数据延迟 */
),
MCBSP_XCR1_RMK(
MCBSP_XCR1_XFRLEN1_OF(0), /* XFRLEN1 = 0,发送帧每帧一字 */
MCBSP_XCR1_XWDLEN1_16BIT /* XWDLEN1 = 5,每字32位 */
),
MCBSP_XCR2_RMK(
MCBSP_XCR2_XPHASE_SINGLE, /* XPHASE = 0,单段帧 */
MCBSP_XCR2_XFRLEN2_OF(0), /* XFRLEN2 = 0,该帧无字 */
MCBSP_XCR2_XWDLEN2_8BIT, /* XWDLEN2 = 5,每字32bit */
MCBSP_XCR2_XCOMPAND_MSB, /* XCOMPAND = 0,无压缩,先传输高位数据 */
MCBSP_XCR2_XFIG_YES, /* XFIG = 0,每个帧同步信号启动一次数据发送 */
MCBSP_XCR2_XDATDLY_0BIT /* XDATDLY = 1,1bit数据延迟 */
),
MCBSP_SRGR1_RMK(
MCBSP_SRGR1_FWID_OF(1), /* 停止模式无效**重要FWID = 0 帧同步信号的脉宽周期数*/
MCBSP_SRGR1_CLKGDV_OF(0) /* 停止模式无效**重要CLKGDV =1 CLKG时钟频率*/
),
MCBSP_SRGR2_RMK(
0,//MCBSP_SRGR2_GSYNC_FREE, /* FREE = 0 内部时钟帧同步 (每FPER+1个CLKG周期产生FSG脉冲时钟同步)*/
MCBSP_SRGR2_CLKSP_RISING, /* 重要CLKSP = 0 CLKS 引脚极性,CLKS上升沿产生CLKG和FSG*/
MCBSP_SRGR2_CLKSM_INTERNAL, /* 重要CLKSM = 1 cpu时钟(叠加93行SCLKME=0)*/
MCBSP_SRGR2_FSGM_DXR2XSR, /* FSGM = 0 发送帧同步模式,如果FXSM=1,当DXR->XSR时,McBSP产生帧同步信号*/
MCBSP_SRGR2_FPER_OF(0) /*重要 FPER = 0 FSG信号帧同步时钟周期数,应该是7*(每个FSG周期为FPER+1个CLKG时钟周期)*/
),
MCBSP_MCR1_DEFAULT,
MCBSP_MCR2_DEFAULT,
MCBSP_PCR_RMK(
MCBSP_PCR_IDLEEN_RESET, /* IDLEEN = 0,当PERIPH域idle时MCBSP保持有效 */
MCBSP_PCR_XIOEN_SP, /* XIOEN = 0,CLKX,FSX,DX,CLKS用作串口引脚 */
MCBSP_PCR_RIOEN_SP, /* RIOEN = 0,CLKX,FSX,DX,CLKS用作串口引脚 */
MCBSP_PCR_FSXM_EXTERNAL, /* FSXM = 0,接收帧同步信号 */
MCBSP_PCR_FSRM_EXTERNAL, /* FSRM = 1,接收帧同步信号由MCBSP提供 */
MCBSP_PCR_CLKXM_INPUT, /* CLKXM = 0,MCBSP作为SPI从设备 */
MCBSP_PCR_CLKRM_OUTPUT, /* CLKRM = 0,外部通过CLKR引脚提供接收时钟(叠加27行DLB=0) */
MCBSP_PCR_SCLKME_NO, /*SCLKME =0,CPU时钟(叠加79行CLKSM=1) */
//MCBSP_PCR_CLKSSTAT_0,
MCBSP_PCR_DXSTAT_0, /* DXSTAT = 0,DX引脚设为低电平 */
//MCBSP_PCR_DRSTAT_0, /*DRSTAT =0,DR引脚设为低电平 */
MCBSP_PCR_FSXP_ACTIVELOW, /* FSXP = 1,发送帧同步信号低有效 */
MCBSP_PCR_FSRP_ACTIVELOW, /* FSRP = 1,接收帧同步信号低有效 */
MCBSP_PCR_CLKXP_FALLING, /* CLKXP = 0,CLKX上升沿发送数据 */
MCBSP_PCR_CLKRP_FALLING /* CLKRP = 1,CLKR上升沿接收数据(CLKR输出数据信号时内部CLKR经过转换后送到CLKR引脚) */
/*低电平无效,有延迟,MCBSP在CLKX上升沿之前半个周期发送数据,在CLKR上升沿接收数据*/
),
MCBSP_RCERA_DEFAULT,
MCBSP_RCERB_DEFAULT,
MCBSP_RCERC_DEFAULT,
MCBSP_RCERD_DEFAULT,
MCBSP_RCERE_DEFAULT,
MCBSP_RCERF_DEFAULT,
MCBSP_RCERG_DEFAULT,
MCBSP_RCERH_DEFAULT,
MCBSP_XCERA_DEFAULT,
MCBSP_XCERB_DEFAULT,
MCBSP_XCERC_DEFAULT,
MCBSP_XCERD_DEFAULT,
MCBSP_XCERE_DEFAULT,
MCBSP_XCERF_DEFAULT,
MCBSP_XCERG_DEFAULT,
MCBSP_XCERH_DEFAULT
};
选择RINT0作为中断,初始化函数如下所示:
void IRQInit(void)
{
/* 设置中断向量入口
修改寄存器IVPH,IVPD,重新定义中断向量表*/
IRQ_setVecs((Uint32)(&VECSTART));
/*禁止可屏蔽中断使能*/
old_intm = IRQ_globalDisable();
/*获取中断标号*/
eventId0 = IRQ_EVT_RINT0;
/* 清除外部中断*/
IRQ_clear(eventId0);
/*设置中断向量地址*/
IRQ_plug(eventId0,&transmit_ready);
/*使能外部中断*/
IRQ_enable(eventId0);
/*使能可屏蔽中断*/
IRQ_globalEnable();
}
中断函数如下:
interrupt void transmit_ready()
{
Uint16 data = 0;
data = slave_rx16();
delay_us(1);
}
Uint16 slave_rx16(void)
{
Uint16 rx = 0;
while(!MCBSP_rrdy(hMcbsp2));
rx=MCBSP_read16(hMcbsp2);
while(!MCBSP_xrdy(hMcbsp2));
MCBSP_write16(hMcbsp2,0x1234);
return rx;
}
利用STM32作为主机循环发送16位数据0x9898(1001 1000 1001 1000),但DSP总会接收到错误数据,如下图所示:

除此之外,诡异的是,断开clkx0与主机的连线时,DSP依然可以完成读数(rx = 0x0),接收数据的寄存器可以在没有主时钟的情况下工作?
希望工程师们能解答我的疑惑,感谢!