我有一些驱动程序代码正在测试中,以便与SSDx1306驱动的OLED屏幕(128x32)(与OLED adafruit型号相同)一起使用。我需要它在debian中运行(我正在使用Linario-4.4.9
)
我遵循了Debian指南中有关如何开始为设备创建文件处理程序的指南,如下所示。 oled.h
中的唯一内容是设备地址(0x3C
)和原型类型。我遵循了在adafruit github上采用的初始化方法(因为我首先在Ardunio上尝试了它们的代码,以确保屏幕实际上可以正常工作)。我相信我可能做错了什么,但我不确定自己做错了什么。我还在下面列出了我的初始化过程。
#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
返回IO故障(#5)。但是,打开设备以获取文件指针似乎从来没有问题。
我有时会得到
ERRNO
超时,但是我已经读到这只是使用协议的I2C设备的一个问题,因为write
期望比I2C给出更快的响应。我还使用
-std=c99 -O0
进行编译,以确保所有内联函数都存在,并确保循环变量可用。如果有人可以指出我的正确方向,可以指出我的方法中存在的缺陷,将不胜感激。谢谢。
编辑
我已经检查了设备树,并正确启用了i2c设备。但是,似乎没有启用任何
i2c_freq
速度。这会导致超时和垃圾数据传输吗? 最佳答案
我已经连接了逻辑分析仪来嗅探I2C线路,看来当我连接了LA时,I2C线路不再起作用,并且ERRNO返回IO故障(#5)。
逻辑分析仪只是一种测量设备。它将捕获的数据转换为时序图,并对您设置的协议进行解码。因此,它将不对任何I2C读写错误负责(直到您的接地和硬件连接正确)。
对于超时问题,您可以尝试减小i2c clock-frequency
或ioctl I2C_TIMEOUT
。
关于c - 用于OLED屏幕的Debian I2C驱动程序不起作用。 [ssd1306],我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/43333649/