你好,
我尝试与传感器进行通信。传感器数据表说我需要编写测量命令。它的值(12或14位)存储在其内存中。要获取此数据,我必须读出2个字节。但每当我尝试用“I2C_EEPROM_READ_DATA(UINT8_T * RD_DATA_PTR,UINT32_T地址,UINT32_T大小)
“我被困在”read_data_single()
“ 在里面 ”WAIT_FOR_RECEIVED_BYTE ();
“循环。似乎DA14580没有收到一些东西。
为* rd_data_ptr.
我有一个指向一个值变量的指针uint32_t mes_val.
.对于我使用的地址i2c_slave_address.
由user_外围_setup.h配置。为尺寸
我用2表示两个字节。
我该怎么办才能收到两个字节?它如何存储?我需要存储在一个变量中的值,该变量大小为2bytes。发送“读取”命令后,传感器发送“most_significant_byte”_ack_“最终_significant_byte”_ack_(也许校验和)_nack_ stop_condition。
为了更好地描述它,我也尝试了i2c_eeprom_read_byte.
获得第一个字节。但这也不起作用。BLE连接也丢失了,我无法重新连接。整个系统仅在“等待循环”中。我必须取消,重置和重新启动系统以再次获得蓝牙。我想通过BLE-命令触发测量,并将1写入特征值。在写入1到CHAR值,应触发测量的触发命令(1byte长度)。之后我想读出将存储在另一个特征中的值(2个字节)。
当一切工作时,我想使用通知更新我的手机上的值与传感器的新值。
谢谢!
1月
设备:
嗨Janfeld,
如果您被困在WAIT_FOR_RECEIVE_BYTE()中,那么580没有从另一边接收任何东西。我不认为这是由于代码实现,而是由于与580和传感器的连接,或者传感器出于某种原因从不响应您发送的命令。在这种情况下,捕获一个逻辑分析器会有很大帮助。
关于BLE,I2C驱动程序轮询I2C接口以获得一些数据,因为没有数据返回驱动程序,因为M0卡在I2C轮询中,因此丢失了驱动程序。您必须捕获I2C总线,并查看580在总线上发送的内容,如果它是正确的(传感器获取他想要的命令),如果传感器的回复。
关于I2C驱动程序的读取过程,您可以使用read_data_single()函数设置要读取的传感器的地址(传感器的某些内部地址)并将大小设置为2.将强制驱动程序要发送两个时钟周期,以便从声明的指定地址中读取数据。然后,驱动程序将放置在指向函数中作为参数传递的指针中收到的数据。
谢谢mt_dialog.
感谢您的回答。我没有关于传感器内部地址的任何类型的信息。与Arduino合作罚款。我将“测量CMD”发送到传感器,在测量时等待一些毫秒,然后发送读取请求。之后,我将读取两个字节的数据。传感器的内部地址没有任何内容。我认为这是这样的:
μC:_____ s___addr + wr_bit___meas_cmd____addr + rd_bit____Ack____ACK___________________________
传感器:______________________Ack__________________________MSB______LSB_____CHECKSUM__________
这是传感器数据表中描述的。没有特定的指针指令地址可用。是否有其他方法来接收这些值?比如一个“听”功能?
我会尽快使用示波器分析传输。但也许你有一个想法我能做什么。
谢谢,
1月
嗨Janfeld,
似乎您必须将meas_cmd作为read_data_single()中的地址参数发送为2,以便执行两个读数。为了使I2C传感器提供一些值,需要了解EMIT和哪些设备或通常做的数据的数据,因此您必须向I2C总线提供您想要的设备的地址要与传感器进行通信并写入传感器(I2C的写命令)您想要哪些值(MEAM_CMD)。因此,当您尝试使用read_data_single()函数访问总线时,i2c将自动提供在I2C模块初始化期间的从设备的地址(在I2C_INIT())和写入位置在从总线上遵循的从地址(MEAS_CMD)之后,执行I2C_SEND_ADDRESS(MEAM_CMD),然后I2C模块将提供从传感器采用两个字节的时钟。我希望我足够清楚。
谢谢mt_dialog.
嗨,它仍然不起作用。我试图将meas_cmd发送为地址参数。
我从传感器的数据表附上了方案。它显示了传感器使用的通信序列。到目前为止,我无法在I2C_EEPROM文件中看到可以提供这种通信流的可能性。
在尝试接收字节时,我仍然会卡住。
在我的理解中,我必须用写函数发送测量命令。我没有内存地址,而是只有传感器的I2C地址。测量后(20-80μs之间的某些东西),我可以发送read-command(i2c_address +读取位)。然后传感器开始发送数据。起初最重要的字节。当收到它时,DA14580必须提供ACK。然后将发送第二个字节(LSB)(然后是来自DA14580的ACK)。校验和将最后发送。
我是否弄错了关于它如何工作的原因,或者是我看不到的东西?
对于通信,两条通信线路必须有上拉。我使用了我在“configurepin(...... input_pullup .....)上设置的内部上拉?那是错的吗?我也尝试过外部上拉。(
configurepin(.....输入.....)
)。也没有工作。谢谢,
1月
嗨Janfeld,
正如我所提到的,如果没有捕获,就没有简单的调试方法,您将不得不检查总线上的信号,仅凭推测我们无法跟踪问题的原因。你必须检查I2C总线上有什么输出。I2C功能当执行一个读将执行写第一次(最后一点I2C奴隶地址设置为0),为了写设备的内部地址,或者一个命令在你的情况中,然后将读取命令(奴隶地址最后位设置为1)为了从传感器开始阅读,您不会在写和读命令之间发出延迟。您可以检查的是传感器I2C的配置,如果它对应于初始化期间I2C模块的设置。但是为了解决这个问题你真的需要捕获总线。
谢谢mt_dialog.
你好,我们又见面了,
我检查了与示波器的通信,发现了
send_i2c_command.
不正常工作。当我想编写一个命令时,它在命令后发送addr + wr_bit后,该命令将一位向右移动。(大多数时候,不是所有的时间..有时它有效)例如:
send_i2c_command(0xF3)
将发送:addr + wr_bit和0x71作为命令。(1000'0000为7位ADDR。+ WR位0和0111'1001而不是1111'0011)。地址是正确的,有时它有效,传感器正在进行工作。但大多数时间它都不工作并发送错误的值。当我尝试阅读某些东西时也一样。我想发送
SEND_I2C_COMMAND (0 x0100)
.它发送:ADDR+WR_BIT(而不是RD_BIT),然后是一些随机值。有时它工作,它发送ADDR+RD_BIT。(10000001)但是大多数时候它不能正常工作。我拥有的另一个问题是,是ACK自动设置或者我必须手动将其设置为Ack第一个字节,以便传感器可以发送第二个字节?
谢谢,
1月
嗨Janfeld,
关于send_i2c_command(),我不认为该命令必须与您所面临的内容进行任何操作,如果有时它有效,另外但我不认为它必须用您的连接做点什么。
当他自动获取收到的数据时,I2C控制器将生成ACK,您不必从代码中执行任何额外的事情。
谢谢mt_dialog.
你好,
我有同样的问题,即在I2C总线上读取多字节时陷入了卡。我从“i2c_eeprom.c”之一修改了I2C-Multi-Read Fucntioin。该功能如下所述。
而问题是,当我试图从传感器(加速计MMA8652)读取192个字节时,程序将停止在“WAIT_FOR_RECEIVED_BYTE();”,其中已经读取的数据的索引总是66(dec)。
并且我从逻辑信号分析仪上得到了通信图,看起来DA14580和加速度计之间的通信是正确的,信号持续了192个字节的周期!当我删除代码“WAIT_FOR_RECEIVED_BYTE”时,整个周期的时间是相同的。根据图表,甚至在I2C-STOP信号之前就有了可用的数据。
因此,似乎DA14580和加速度计之间的“连接问题”不是正确的原因。
希望你的帮助。
void i2c_readmultibyte(uint8_t regaddr,uint8_t data [],uint8_t大小)
{
uint8_t索引;
WAIT_UNTIL_I2C_FIFO_IS_EMPTY ();
wait_until_no_master_actity();
//注册地址阅读
SEND_I2C_COMMAND (regAddr);
for(index = 0;索引{
WAIT_WHILE_I2C_FIFO_IS_FULL ();//等待TX FIFO已满
SEND_I2C_COMMAND (0 x0100);//设置读访问次数
}
//数据阅读{
for(index = 0;索引
WAIT_FOR_RECEIVED_BYTE ();
数据[index] = (uint8_t)(GetWord16(I2C_DATA_CMD_REG) & 0xFF);
}
}
并初始化I2C Contorler:
i2cinitparam_t param;
param.periadrr = 0x1d;
参数。addressMode = I2C_7BIT_ADDR;
param.speedmode = i2c_fast;
void I2C_Init(const i2cInitParam_t *param)
{
setbits16(clk_per_reg,i2c_enable,1);//为i2c启用时钟
setword16(i2c_enable_reg,0x0);//禁用I2C控制器
setword16(i2c_con_reg,i2c_master_mode | i2c_slave_disable | i2c_restart_en);//从站被禁用
setBits16(i2c_con_reg,i2c_speed,param-> speedmode);//设置速度
SetBits16 (I2C_CON_REG I2C_10BITADDR_MASTER, param - > addressMode);//设置寻址模式
setword16(i2c_tar_reg,param-> periadrr&0x3ff);//设置从设备地址
setword16(i2c_enable_reg,0x1);//启用I2C控制器
而((getword16(i2c_status_reg)&0x20)!= 0);//等待I2C主FSM闲置
}
你好,
我有同样的问题,即在I2C总线上读取多字节时陷入了卡。我从“i2c_eeprom.c”之一修改了I2C-Multi-Read Fucntioin。该功能如下所述。
而问题是,当我试图从传感器(加速计MMA8652)读取192个字节时,程序将停止在“WAIT_FOR_RECEIVED_BYTE();”,其中已经读取的数据的索引总是66(dec)。
并且我从逻辑信号分析仪上得到了通信图,看起来DA14580和加速度计之间的通信是正确的,信号持续了192个字节的周期!当我删除代码“WAIT_FOR_RECEIVED_BYTE”时,整个周期的时间是相同的。根据图表,甚至在I2C-STOP信号之前就有了可用的数据。
因此,似乎DA14580和加速度计之间的“连接问题”不是正确的原因。
希望你的帮助。
void i2c_readmultibyte(uint8_t regaddr,uint8_t data [],uint8_t大小)
{
uint8_t索引;
WAIT_UNTIL_I2C_FIFO_IS_EMPTY ();
wait_until_no_master_actity();
//注册地址阅读
SEND_I2C_COMMAND (regAddr);
for(index = 0;索引{
WAIT_WHILE_I2C_FIFO_IS_FULL ();//等待TX FIFO已满
SEND_I2C_COMMAND (0 x0100);//设置读访问次数
}
//数据阅读{
for(index = 0;索引
WAIT_FOR_RECEIVED_BYTE ();
数据[index] = (uint8_t)(GetWord16(I2C_DATA_CMD_REG) & 0xFF);
}
}
并初始化I2C Contorler:
i2cinitparam_t param;
param.periadrr = 0x1d;
参数。addressMode = I2C_7BIT_ADDR;
param.speedmode = i2c_fast;
void I2C_Init(const i2cInitParam_t *param)
{
setbits16(clk_per_reg,i2c_enable,1);//为i2c启用时钟
setword16(i2c_enable_reg,0x0);//禁用I2C控制器
setword16(i2c_con_reg,i2c_master_mode | i2c_slave_disable | i2c_restart_en);//从站被禁用
setBits16(i2c_con_reg,i2c_speed,param-> speedmode);//设置速度
SetBits16 (I2C_CON_REG I2C_10BITADDR_MASTER, param - > addressMode);//设置寻址模式
setword16(i2c_tar_reg,param-> periadrr&0x3ff);//设置从设备地址
setword16(i2c_enable_reg,0x1);//启用I2C控制器
而((getword16(i2c_status_reg)&0x20)!= 0);//等待I2C主FSM闲置
}
我有一个为我工作的解决方案,但我认为这可能不足以一行的大量字节。但也许你可以尝试..
我通过删除
WAIT_FOR_RECEIVED_BYTE ();
并尝试直接获取该值并通过指针将其保存到数组。我的函数现在看起来像这样:
void i2c_cust_read_bytes(uint8_t * val_ptr)
{
val_ptr [0] = 0x00;
val_ptr [1] = 0x00;
wait_until_no_master_actity();
SEND_I2C_COMMAND (0 x0100);
WAIT_WHILE_I2C_FIFO_IS_FULL ();
val_ptr [1] = getword16(i2c_data_cmd_reg);
SEND_I2C_COMMAND (0 x0100);
WAIT_WHILE_I2C_FIFO_IS_FULL ();
val_ptr [0] = getword16(i2c_data_cmd_reg)&0xfc;
}
这对我很有效。我得到数组第二个元素的第一个值的原因只是我想让它在那里。您可以选择要指向哪个元素。
我想应该是这样的:
起初:没有主活动。
然后发送“read_cmd”..等待直到FIFO为空,以接收新字节(在将地址+读取位和0x00写入I2C_DATA_CMD_REG时清除)。
然后保存到您想要的变量。再次发送一个READ_CMD,它将覆盖I2C_DATA_CMD_REG中保存的值,并对您希望在一行中接收的每个字节重复该操作。
对于它可能工作的两个值,但我不知道是否可以在没有错误或更改位的行中保存以接收多个或两个字节。
可悲的是,我仍然不知道为什么wait_for_received_byte()不适合我..
希望我能帮助你一点。
干杯,
1月
你好,
你是否试过使用I2C驱动程序(i2c_eeprom_read_data)。捕捉你的高度我看到没有停止条件从580年为了读取数据(FIFO满-32时钟周期emmited和32字节的数据已收到,所以proccessor应该开始阅读FIFO和把数据从I2C FIFO缓冲区),当FIFO满时,你正在读取数据吗?WAIT_FOR_RECEIVED_BYTE检查I2C_RX_FLR_REG,这个寄存器包含接收到的有效数据,并且位于FIFO中,如果在FIFO中没有数据,它将被卡在while中,因为它被指示读取。在代码中,输入的大小是多少?如果你调用的功能,你提到超过32,那么你的FIFO是满的,在FIFO的数据是重写的下一个传入的数据。如果你检查I2C_eeprom司机你会看到i2c_eeprom_read_data是阅读的数据使用read_data_single eepm在32的倍数()函数和当FIFO满32字节I2C模块将停止时钟读取32个字节的数据。
Thansk mt_dialog.
你好。
我有一个同样的问题并得到解决方案。
您可以使用此代码读取两个字节。
键是I2C_CON_REG的I2C_RESTART_EN位禁用。
void init_i2c(void)
{
setbits16(clk_per_reg,i2c_enable,1);//为i2c启用时钟
setword16(i2c_enable_reg,0x0);//禁用I2C控制器
//这是关键解决方案。
// setword16(i2c_con_reg,i2c_master_mode | i2c_slave_disable | i2c_restart_en);//从站被禁用
setword16(i2c_con_reg,i2c_master_mode | i2c_slave_disable);
setBits16(i2c_con_reg,i2c_speed,i2c_standard);//设置速度
setbits16(i2c_con_reg,i2c_10bitaddr_master,i2c_7bit_addr);//设置寻址模式
// I2C设备地址设置。
setword16(i2c_tar_reg,地址);//设置从设备地址
setword16(i2c_enable_reg,0x1);//启用I2C控制器
而((getword16(i2c_status_reg)&0x20)!= 0);//等待I2C主FSM闲置
}
无符号短readshorti2c(无效)
{
无符号短V,V1,V2;
WAIT_WHILE_I2C_FIFO_IS_FULL ();//设置R/W位为1 (read access)
SEND_I2C_COMMAND (0 x0100);//设置次次读取访问。所以2次来电
WAIT_WHILE_I2C_FIFO_IS_FULL ();
SEND_I2C_COMMAND (0 x0100);
WAIT_FOR_RECEIVED_BYTE ();//等待收到的数据
v1 = 0xff&getword16(i2c_data_cmd_reg);//获得收到的字节,MSB
WAIT_FOR_RECEIVED_BYTE ();//等待收到的数据
v2 =(0xff&getword16(i2c_data_cmd_reg));//获得收到的字节,LSB
V = v1<<8;
ARCH_PRINTF(“\ r \ nv%04x:%04x =%04x(%d)”,v1,v2,v,v);
返回v;
}
谢谢你。