DHT11模块调试问题

dth11模块调试发现的问题:

  • gpio引脚输出时不该再设置方向。
static void dht11_start(void)
{
    //此处bug
    gpiod_direction_output(dht11_data_pin, GPIOD_OUT_LOW);
    mdelay(18);
    gpiod_direction_output(dht11_data_pin, GPIOD_OUT_HIGH);
    udelay(40);    
    gpiod_direction_input(dht11_data_pin);
}
static void dth11_start(void){
    //启动信号 
    gpiod_direction_output(dth11_pin,GPIOD_OUT_HIGH);
    udelay(2);
    gpiod_set_value(dth11_pin,0);
    mdelay(18);//18ms的低脉冲。40us的高脉冲
    gpiod_set_value(dth11_pin,1);
    udelay(40);
    gpio_direction_input(dth11_pin);
    udelay(2);
}

 

 上述第一张图是错误示范;(重复设置输出方向,会清空方向寄存器,此时引脚为输入引脚,不知道其电平是高是低)。  

  • 当输出高脉冲的时候应该等待高脉冲完成。

imx6ull每读取一次gpio,大约耗时1us,udelay一次需要1.6us;使用 ktime_get_boot_ns()准确。

static int dht11_read_byte(unsigned char *buf)
{
    int i;
    int us = 0;
    unsigned char data = 0;
    int timeout_us = 200;
    u64 pre, last, last2;
    int ns;
    int j;
    
    for (i = 0; i <8; i++)
    {
        /* 现在是低电平 */
        /* 等待高电平 */
        timeout_us = 400;
        us = 0;
        while (!gpiod_get_value(dht11_data_pin) && --timeout_us)
        {
            udelay(1);
            us++;
        }
        if (!timeout_us)
        {
            printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
            return -1;
        }
        us_low_array[us_low_index++] = us;

        /* 现在是高电平 */
        /* 等待低电平,累加高电平的时间 */
        timeout_us = 20000000;
        us = 0;
        
        //last2 = ktime_get_boot_ns();
        //for (j = 0; j < 23; j++)
        udelay(40);

        if (gpiod_get_value(dht11_data_pin))
        {
            /* get bit 1 */
            data = (data << 1) | 1;

            /* 当前位的高电平未结束, 等待 */
            timeout_us = 400;
            us = 0;
            while (gpiod_get_value(dht11_data_pin) && --timeout_us)
            {
                udelay(1);
                us++;
            }
            if (!timeout_us)
            {
                printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);
                return -1;
            }
            
        }
        else
        {
            /* get bit 0 */
            data = (data << 1) | 0;
        }
    }

    *buf = data;
    return 0;
}
  •  使用中断方式,丢失中断的问题:得到的中断次数不一定都是82次;当ack信号准备好之后,再request_irq时,可能会漏一些中断,在解析数据的时候,可以计算三次校验码,(imx6ull问题比较严重,stm32mp157效果较好,双核)
static int dht11_parse_data(char* data){
  int i,j,m = 0;
  int offset = 0;
  for (offset = 0; offset <= 2; offset++){
    m = offset;//尝试计算三次,因为可能会计算进去ack时候的中断
    for(i= 0; i < 5; i ++){
      data[i]=0;
      for(j = 0;j < 8; j++){
        data[i] <<= 1;
        if(dht11_edge_time[m+1] - dht11_edge_time[m] >= 40000){
          data[i] |= 1;
        }
        m += 2;
      }
    }
    if(data[4] == data[0] + data[1] + data[2] + data[3]){
      return 0;
    }
  }
  return -1;
}

 

posted @ 2022-11-02 11:57  QianFa01  阅读(337)  评论(0)    收藏  举报