commit | author | age
|
14da47
|
1 |
/********************************************************************************* |
G |
2 |
* Copyright: (C) 2023 LingYun IoT System Studio |
|
3 |
* All rights reserved. |
|
4 |
* |
|
5 |
* Filename: ds18b20.c |
|
6 |
* Description: This file is temperature sensor DS18B20 source code |
|
7 |
* |
|
8 |
* Version: 1.0.0(2023/8/10) |
|
9 |
* Author: Guo Wenxue <guowenxue@gmail.com> |
|
10 |
* ChangeLog: 1, Release initial version on "2023/8/10 12:13:26" |
|
11 |
* |
|
12 |
* Pin connection: |
|
13 |
* |
|
14 |
* DS18B20 Module IGKBoard |
|
15 |
* VCC <-----> 3.3V |
|
16 |
* DQ <-----> #Pin7(GPIO1_IO18) |
|
17 |
* GND <-----> GND |
|
18 |
* |
|
19 |
* /run/media/mmcblk1p1/config.txt: |
|
20 |
* |
|
21 |
* # Eanble 1-Wire overlay |
|
22 |
* dtoverlay_w1=yes |
|
23 |
* |
|
24 |
********************************************************************************/ |
|
25 |
|
|
26 |
/* 在C语言编程时,一般系统的头文件用<xxx.h>,我们自己写的头文件则用"zzz.h" */ |
|
27 |
#include <stdio.h> |
|
28 |
#include <stdlib.h> |
|
29 |
#include <unistd.h> |
|
30 |
#include <fcntl.h> |
|
31 |
#include <dirent.h> |
|
32 |
#include <string.h> |
|
33 |
#include <time.h> |
|
34 |
#include <errno.h> |
|
35 |
|
|
36 |
/* 在C语言编程中,函数应该先定义再使用,如果函数的定义在函数调用后面,应该前向声明。*/ |
|
37 |
int ds18b20_get_temperature(float *temp); |
|
38 |
|
|
39 |
int main(int argc, char *argv[]) |
|
40 |
{ |
|
41 |
float temp; /* 温度值有小数位,所以使用浮点数 */ |
|
42 |
|
|
43 |
/* 1,在Linux下做C语言编程时,函数返回值一般是0表示成功,<0表示失败,我们也遵循这个规约; |
|
44 |
* 2,但函数调用只能有一个返回值,所以这里的采样函数只能通过指针来返回采样的温度值; |
|
45 |
* 3,因为要在ds18b20_get_temperature()函数中修改main()中temp的值,所以这里传&temp; |
|
46 |
*/ |
|
47 |
if( ds18b20_get_temperature(&temp) < 0 ) |
|
48 |
{ |
|
49 |
printf("ERROR: ds18b20 get temprature failure\n"); |
|
50 |
return 1; |
|
51 |
} |
|
52 |
|
|
53 |
/* 打印DS18B20的采样温度值,因为℃是非ASCII打印字符,所以这里用 'C 代替 */ |
|
54 |
printf("DS18B20 get temperature: %f 'C\n", temp); |
|
55 |
|
|
56 |
return 0; |
|
57 |
} |
|
58 |
|
|
59 |
/* |
|
60 |
* 函数说明: 该函数用来使用 DS18B20 温度传感器采样获取当前的温度值; |
|
61 |
* 参数说明: $temp: 通过指针返回DS18B20的采样温度 |
|
62 |
* 返回说明: ==0 表示成功, <0 表示失败 |
|
63 |
*/ |
|
64 |
int ds18b20_get_temperature(float *temp) |
|
65 |
{ |
|
66 |
const char *w1_path = "/sys/bus/w1/devices/"; |
|
67 |
char ds_path[50]; /* DS18B20 采样文件路径 */ |
|
68 |
char chip[20]; /* DS18B20 芯片序列号文件名 */ |
|
69 |
char buf[128]; /* read() 读数据存储 buffer */ |
|
70 |
DIR *dirp; /* opendir()打开的文件夹句柄 */ |
|
71 |
struct dirent *direntp; /* readdir()读文件夹内容时的目录项*/ |
|
72 |
int fd =-1; /* open()打开文件的文件描述符 */ |
|
73 |
char *ptr; /* 一个字符指针,用来字符串处理 */ |
|
74 |
int found = 0; /* 是否找到DS18B20的标志,默认设置为没找到(0) */ |
|
75 |
int rv = 0; /* 函数返回值,默认设置为成功返回(0) */ |
|
76 |
|
|
77 |
|
|
78 |
/* 在C语言编程时,进入函数的第一件事应该进行函数参数的合法性检测,检查参数非法输入。 |
|
79 |
* 否则调用者"不小心"通过 $temp 传入一个空指针,下面的代码就有可能出现段错误。 |
|
80 |
*/ |
|
81 |
if( !temp ) |
|
82 |
{ |
|
83 |
return -1; |
|
84 |
} |
|
85 |
|
|
86 |
/* 打开 "/sys/bus/w1/devices/" 文件夹,如果打开失败则打印错误信息并退出。 */ |
|
87 |
if((dirp = opendir(w1_path)) == NULL) |
|
88 |
{ |
|
89 |
printf("opendir error: %s\n", strerror(errno)); |
|
90 |
return -2; |
|
91 |
} |
|
92 |
|
|
93 |
/* 1, 因为文件夹下可能有很多文件,所以这里使用while()循环读取/sys/bus/w1/devices/ |
|
94 |
* 文件夹下的所有目录项,其中 direntp->d_name 就是目录里的每个文件/文件夹的文件名。 |
|
95 |
* |
|
96 |
* 2, 接下来我们使用 strstr() 函数判断文件名中是否包含 "28-",如果找到则将完整的 |
|
97 |
* 文件名通过strcpy()函数保存到 chip 中;并设置 found 标志为1,跳出循环。 |
|
98 |
*/ |
|
99 |
while((direntp = readdir(dirp)) != NULL) |
|
100 |
{ |
|
101 |
if(strstr(direntp->d_name,"28-")) |
|
102 |
{ |
|
103 |
/* find and get the chipset SN filename */ |
|
104 |
strcpy(chip,direntp->d_name); |
|
105 |
found = 1; |
|
106 |
break; |
|
107 |
} |
|
108 |
} |
|
109 |
|
|
110 |
/* 文件夹打开用完后,要记得第一时间关闭 */ |
|
111 |
closedir(dirp); |
|
112 |
|
|
113 |
/* found在定义时初始化为0,如果上面的代码没有找到 "28-" 文件名则其值依然为0,否则将会被 |
|
114 |
* 设置为1。如果 found 的值为0的话,则打印错误信息并返回相应的错误码-3. |
|
115 |
*/ |
|
116 |
if( !found ) |
|
117 |
{ |
|
118 |
printf("Can not find ds18b20 in %s\n", w1_path); |
|
119 |
return -3; |
|
120 |
} |
|
121 |
|
|
122 |
/* 使用snprintf()函数生成完整路径/sys/bus/w1/devices/28-xxxxx/w1_slave |
|
123 |
* 并保存到 ds_path 中。 |
|
124 |
*/ |
|
125 |
snprintf(ds_path, sizeof(ds_path), "%s/%s/w1_slave", w1_path, chip); |
|
126 |
|
|
127 |
/* 接下来打开 DS18B20 的采样文件,如果失败则返回相应的错误码-4。 */ |
|
128 |
if( (fd=open(ds_path, O_RDONLY)) < 0 ) |
|
129 |
{ |
|
130 |
printf("open %s error: %s\n", ds_path, strerror(errno)); |
|
131 |
return -4; |
|
132 |
} |
|
133 |
|
|
134 |
/* 读取文件中的内容将会触发 DS18B20温度传感器采样,这里读取文件内容保存到buf中 */ |
|
135 |
if(read(fd, buf, sizeof(buf)) < 0) |
|
136 |
{ |
|
137 |
printf("read %s error: %s\n", ds_path, strerror(errno)); |
|
138 |
|
|
139 |
/* 1, 这里不能直接调用 return -5 直接返回,否则的话前面open()打开的文件描述符就没有关闭。 |
|
140 |
* 这里设置 rv 为错误码-5,通过 goto 语句跳转到函数后面统一进行错误处理。 |
|
141 |
* |
|
142 |
* 2, 在C语言编程时我们应该慎用goto语句进行"随意"的跳转,因为它会降低代码的可读性。但这里是 |
|
143 |
* goto语句的一个非常典型应用,我们经常会用它来对错误进行统一的处理。 |
|
144 |
* |
|
145 |
* 3,goto后面的cleanup为标号,它在下面的代码中定义。 |
|
146 |
*/ |
|
147 |
rv = -5; |
|
148 |
goto cleanup; |
|
149 |
} |
|
150 |
|
|
151 |
/* 采样温度值是在字符串"t="后面,这里我们从buf中找到"t="字符串的位置并保存到ptr指针中 */ |
|
152 |
ptr = strstr(buf, "t="); |
|
153 |
if( !ptr ) |
|
154 |
{ |
|
155 |
printf("ERROR: Can not get temperature\n"); |
|
156 |
rv = -6; |
|
157 |
goto cleanup; |
|
158 |
} |
|
159 |
|
|
160 |
/* 因为此时ptr是指向 "t="字符串的地址(即't'的地址),那跳过2个字节(t=)后面的就是采样温度值 */ |
|
161 |
ptr+=2; |
|
162 |
|
|
163 |
/* 接下来我们使用 atof() 函数将采样温度值字符串形式,转化成 float 类型。*/ |
|
164 |
*temp = atof(ptr)/1000; |
|
165 |
|
|
166 |
/* 1,在这里我们对函数返回进行集中处理,其中 cleanup 为 goto 语句的标号; |
|
167 |
* 2,在函数退出时,我们应该考虑清楚在前面的代码中做了哪些事,这些事是否需要进行反向操作。如 |
|
168 |
* 打开的文件或文件夹是否需要关闭,malloc()分配的内存是否需要free()等。 |
|
169 |
* 3, 在最开始我们定义rv并赋初值为0(表示成功)是有原因的,如果前面的代码任何一个地方出现错误, |
|
170 |
* 则会将rv的值修改为相应的错误码,否则rv的值将始终为0(即没有错误发生),这里将统一返回。 |
|
171 |
*/ |
|
172 |
cleanup: |
|
173 |
close(fd); |
|
174 |
return rv; |
|
175 |
} |