我有一些驱动程序代码正在测试中,以便与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/

10-13 08:38