用于 OLED 屏幕的 Debian I2C 驱动程序不工作。 [ssd1306]
Debian I2C driver for OLED screen not working. [ssd1306]
我有一些正在测试的驱动程序代码,用于 SSD1306 驱动的 128x32 OLED 屏幕(与 OLED adafruit 型号相同)。我需要在 debian 中 运行(我正在使用 Linario-4.4.9
)
我已按照 Debian 指南了解如何开始为设备创建文件处理程序,如下所示。 oled.h
中唯一的东西是设备地址 (0x3C
) 和原型类型。我遵循了在 adafruit github 上采用的初始化方法(因为我首先在 Ardunio 上尝试了他们的代码以确保屏幕确实有效)。我相信我可能做错了什么,但我不完全确定我做错了什么。我还在下面列出了我的 init 进程。
#include <errno.h>
#include <stdint.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <linux/i2c-dev.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include "oled.h"
int oled;
int lcd_driver_init(void)
{
///< Begin the init proc.
int dloc = open("/dev/i2c-1", O_RDWR);
if (dloc < 0 )
{
fprintf(stderr, "Error opening i2c device\n");
return -1;
}
if(ioctl(dloc, I2C_SLAVE, SCR_ADDR) < 0)
{
fprintf(stderr, "Error in ioctl. Errno :%i\n",errno);
return -2;
}
oled = dloc;
fprintf(stderr, "init success, device open and local\n");
return EXIT_SUCCESS;
}
int oled_command( uint8_t cmd)
{
char command[2]= {0};
command[1] = cmd;
int check = (write(oled, command, 2));
return check;
}
void oled_cmd_start()
{
int check = (write(oled, 0x00, sizeof(uint8_t)));
if(check<0)
fprintf(stderr, "Errno set:: %i\n", errno);
return;
}
void oled_data_start()
{
uint8_t _data_start_[1] ={ 0x40 };
int check = (write(oled, _data_start_, sizeof(uint8_t)));
if(check<0)
fprintf(stderr, "Errno set oled_data_start:: %i\n", errno);
return;
}
int oled_data (uint8_t xmit)
{
int check = (write(oled, &xmit, (sizeof(uint8_t))));
if(check<0)
fprintf(stderr, "Errno set oled_data:: %i\n", errno);
return check;
}
初始化进程
void sendcommand(unsigned char payload)
{
oled_data(0x00); //Control Byte - Command
oled_data(payload); //payload
}
void lcd_init(void)
{
sendcommand(0xAE);//--Set Display off
sendcommand(0x00);//--set low column address
sendcommand(0x10);//--set high column address
sendcommand(0x81);//--set contrast control register
sendcommand(0x7f);
sendcommand(0xa1);//--set segment re-map 95 to 0
sendcommand(0xA6);//--set normal display
sendcommand(0xa8);//--set multiplex ratio(1 to 16)
sendcommand(0x1f);//--duty 1/32
sendcommand(0xd3);//--set display offset
sendcommand(0x00);//--not offset
sendcommand(0xd5);//--set display clock divide ratio/oscillator frequency
sendcommand(0xf0);//--set divide ratio
sendcommand(0xd9);//--set pre-charge period
sendcommand(0x22);
sendcommand(0xda);//--set com pins hardware configuration
sendcommand(0x02);//disable left/right remap and set for sequential
sendcommand(0xdb);//--set vcomh
sendcommand(0x49);//--0.83*vref
sendcommand(0x8d);//--set DC-DC enable
sendcommand(0x14);//
sendcommand(0xAF);//--turn on oled panel
sendcommand(0xA4);//--Entire Display ON
}
在此之后,我交替发送 0xFF
以尝试在屏幕上制作条纹。唯一出现的是随机像素。没有连贯性。
我已经连接了一个逻辑分析仪来嗅探 I2C 线路,当我连接 LA 时,I2C 线路似乎不再起作用并且 ERRNO
returns IO 故障 (# 5).
但是,打开设备以获取文件指针似乎从来没有问题。
我有时会收到 ERRNO
超时,但我读到这只是使用协议的 I2C 设备的问题,因为 write
期望比 I2C 可能给出的响应更快。
我也在编译 -std=c99 -O0
以确保所有内联函数都存在,并确保循环变量可用。
如果有人能指出我正确的方向或者指出我方法中的一些缺陷,我们将不胜感激。谢谢。
编辑
我检查了设备树,i2c 设备已正确启用。但是 i2c_freq
速度中的 none 似乎已启用。这会导致超时和垃圾数据传输吗?
I have connected a logic analyzer to sniff the I2C lines, and it appears that when I have the LA connected, the I2C lines no longer function and ERRNO returns an IO fault (#5).
逻辑分析仪只是一种测量设备。它将捕获的数据转换为时序图,解码您设置的协议。因此,它不会对任何 I2C 读写错误负责(直到您的接地和 h/w 连接正确)。
对于超时问题,您可以尝试减少 i2c clock-frequency
或 ioctl I2C_TIMEOUT
。
事实证明,SOM 有一个内部调节器,用于将 I2C 线路设为 1V8,而 SSD1306 芯片 运行 为 3V3,导致信息处理不当。 SOM 上未记录此行为。
在设计中应用电平转换芯片可以实现正确的通信。
如果有人遇到同样的问题,请检查您的原理图以了解电压不匹配水平。
我有一些正在测试的驱动程序代码,用于 SSD1306 驱动的 128x32 OLED 屏幕(与 OLED adafruit 型号相同)。我需要在 debian 中 运行(我正在使用 Linario-4.4.9
)
我已按照 Debian 指南了解如何开始为设备创建文件处理程序,如下所示。 oled.h
中唯一的东西是设备地址 (0x3C
) 和原型类型。我遵循了在 adafruit github 上采用的初始化方法(因为我首先在 Ardunio 上尝试了他们的代码以确保屏幕确实有效)。我相信我可能做错了什么,但我不完全确定我做错了什么。我还在下面列出了我的 init 进程。
#include <errno.h>
#include <stdint.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <linux/i2c-dev.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include "oled.h"
int oled;
int lcd_driver_init(void)
{
///< Begin the init proc.
int dloc = open("/dev/i2c-1", O_RDWR);
if (dloc < 0 )
{
fprintf(stderr, "Error opening i2c device\n");
return -1;
}
if(ioctl(dloc, I2C_SLAVE, SCR_ADDR) < 0)
{
fprintf(stderr, "Error in ioctl. Errno :%i\n",errno);
return -2;
}
oled = dloc;
fprintf(stderr, "init success, device open and local\n");
return EXIT_SUCCESS;
}
int oled_command( uint8_t cmd)
{
char command[2]= {0};
command[1] = cmd;
int check = (write(oled, command, 2));
return check;
}
void oled_cmd_start()
{
int check = (write(oled, 0x00, sizeof(uint8_t)));
if(check<0)
fprintf(stderr, "Errno set:: %i\n", errno);
return;
}
void oled_data_start()
{
uint8_t _data_start_[1] ={ 0x40 };
int check = (write(oled, _data_start_, sizeof(uint8_t)));
if(check<0)
fprintf(stderr, "Errno set oled_data_start:: %i\n", errno);
return;
}
int oled_data (uint8_t xmit)
{
int check = (write(oled, &xmit, (sizeof(uint8_t))));
if(check<0)
fprintf(stderr, "Errno set oled_data:: %i\n", errno);
return check;
}
初始化进程
void sendcommand(unsigned char payload)
{
oled_data(0x00); //Control Byte - Command
oled_data(payload); //payload
}
void lcd_init(void)
{
sendcommand(0xAE);//--Set Display off
sendcommand(0x00);//--set low column address
sendcommand(0x10);//--set high column address
sendcommand(0x81);//--set contrast control register
sendcommand(0x7f);
sendcommand(0xa1);//--set segment re-map 95 to 0
sendcommand(0xA6);//--set normal display
sendcommand(0xa8);//--set multiplex ratio(1 to 16)
sendcommand(0x1f);//--duty 1/32
sendcommand(0xd3);//--set display offset
sendcommand(0x00);//--not offset
sendcommand(0xd5);//--set display clock divide ratio/oscillator frequency
sendcommand(0xf0);//--set divide ratio
sendcommand(0xd9);//--set pre-charge period
sendcommand(0x22);
sendcommand(0xda);//--set com pins hardware configuration
sendcommand(0x02);//disable left/right remap and set for sequential
sendcommand(0xdb);//--set vcomh
sendcommand(0x49);//--0.83*vref
sendcommand(0x8d);//--set DC-DC enable
sendcommand(0x14);//
sendcommand(0xAF);//--turn on oled panel
sendcommand(0xA4);//--Entire Display ON
}
在此之后,我交替发送 0xFF
以尝试在屏幕上制作条纹。唯一出现的是随机像素。没有连贯性。
我已经连接了一个逻辑分析仪来嗅探 I2C 线路,当我连接 LA 时,I2C 线路似乎不再起作用并且 ERRNO
returns IO 故障 (# 5).
但是,打开设备以获取文件指针似乎从来没有问题。
我有时会收到 ERRNO
超时,但我读到这只是使用协议的 I2C 设备的问题,因为 write
期望比 I2C 可能给出的响应更快。
我也在编译 -std=c99 -O0
以确保所有内联函数都存在,并确保循环变量可用。
如果有人能指出我正确的方向或者指出我方法中的一些缺陷,我们将不胜感激。谢谢。
编辑
我检查了设备树,i2c 设备已正确启用。但是 i2c_freq
速度中的 none 似乎已启用。这会导致超时和垃圾数据传输吗?
I have connected a logic analyzer to sniff the I2C lines, and it appears that when I have the LA connected, the I2C lines no longer function and ERRNO returns an IO fault (#5).
逻辑分析仪只是一种测量设备。它将捕获的数据转换为时序图,解码您设置的协议。因此,它不会对任何 I2C 读写错误负责(直到您的接地和 h/w 连接正确)。
对于超时问题,您可以尝试减少 i2c clock-frequency
或 ioctl I2C_TIMEOUT
。
事实证明,SOM 有一个内部调节器,用于将 I2C 线路设为 1V8,而 SSD1306 芯片 运行 为 3V3,导致信息处理不当。 SOM 上未记录此行为。
在设计中应用电平转换芯片可以实现正确的通信。
如果有人遇到同样的问题,请检查您的原理图以了解电压不匹配水平。