• Home
  • History
  • Annotate
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* i2ctools.c - i2c tools
2  *
3  * Copyright 2018 The Android Open Source Project
4  *
5  * https://www.kernel.org/doc/Documentation/i2c/
6  *
7  * Note: -y must have the same value in each toy for `confirm`.
8  *
9  * TODO: i2cdetect -q/-r and the "auto" mode?
10  * TODO: i2cdump non-byte modes, -r FIRST-LAST?
11  * TODO: i2cget non-byte modes? default to current read address?
12  * TODO: i2cset -r? -m MASK? c/s modes, p mode modifier?
13 
14 USE_I2CDETECT(NEWTOY(i2cdetect, ">3aFly", TOYFLAG_USR|TOYFLAG_BIN))
15 USE_I2CDUMP(NEWTOY(i2cdump, "<2>2fy", TOYFLAG_USR|TOYFLAG_BIN))
16 USE_I2CGET(NEWTOY(i2cget, "<3>3fy", TOYFLAG_USR|TOYFLAG_BIN))
17 USE_I2CSET(NEWTOY(i2cset, "<4fy", TOYFLAG_USR|TOYFLAG_BIN))
18 
19 config I2CDETECT
20   bool "i2cdetect"
21   default y
22   help
23     usage: i2cdetect [-ary] BUS [FIRST LAST]
24     usage: i2cdetect -F BUS
25     usage: i2cdetect -l
26 
27     Detect i2c devices.
28 
29     -a	All addresses (0x00-0x7f rather than 0x03-0x77)
30     -F	Show functionality
31     -l	List all buses
32     -r	Probe with SMBus Read Byte
33     -y	Answer "yes" to confirmation prompts (for script use)
34 
35 config I2CDUMP
36   bool "i2cdump"
37   default y
38   help
39     usage: i2cdump [-fy] BUS CHIP
40 
41     Dump i2c registers.
42 
43     -f	Force access to busy devices
44     -y	Answer "yes" to confirmation prompts (for script use)
45 
46 config I2CGET
47   bool "i2cget"
48   default y
49   help
50     usage: i2cget [-fy] BUS CHIP ADDR
51 
52     Read an i2c register.
53 
54     -f	Force access to busy devices
55     -y	Answer "yes" to confirmation prompts (for script use)
56 
57 config I2CSET
58   bool "i2cset"
59   default y
60   help
61     usage: i2cset [-fy] BUS CHIP ADDR VALUE... MODE
62 
63     Write an i2c register. MODE is b for byte, w for 16-bit word, i for I2C block.
64 
65     -f	Force access to busy devices
66     -y	Answer "yes" to confirmation prompts (for script use)
67 */
68 
69 #define FOR_i2cdetect
70 #include "toys.h"
71 
72 #include <linux/i2c.h>
73 #include <linux/i2c-dev.h>
74 
confirm(const char * fmt,...)75 printf_format static void confirm(const char *fmt, ...)
76 {
77   va_list va;
78 
79   if (toys.optflags & FLAG_y) return;
80 
81   va_start(va, fmt);
82   vfprintf(stderr, fmt, va);
83   va_end(va);
84   if (!yesno(1)) error_exit("Exiting");
85 }
86 
i2c_open(int bus,int slave,int chip)87 static int i2c_open(int bus, int slave, int chip)
88 {
89   int fd;
90 
91   snprintf(toybuf, sizeof(toybuf), "/dev/i2c-%d", bus);
92   fd = xopen(toybuf, O_RDONLY);
93   if (slave) xioctl(fd, slave, (void *)(long)chip);
94   return fd;
95 }
96 
i2c_get_funcs(int bus)97 static unsigned long i2c_get_funcs(int bus)
98 {
99   int fd = i2c_open(bus, 0, 0);
100   unsigned long result;
101 
102   xioctl(fd, I2C_FUNCS, &result);
103   close(fd);
104   return result;
105 }
106 
i2c_read_byte(int fd,int addr,int * byte)107 static int i2c_read_byte(int fd, int addr, int *byte)
108 {
109   struct i2c_smbus_ioctl_data ioctl_data;
110   union i2c_smbus_data data;
111 
112   memset(&data, 0, sizeof(data));
113   ioctl_data.read_write = I2C_SMBUS_READ;
114   ioctl_data.size = I2C_SMBUS_BYTE_DATA;
115   ioctl_data.command = addr;
116   ioctl_data.data = &data;
117   if (ioctl(fd, I2C_SMBUS, &ioctl_data) == -1) return -1;
118   *byte = data.byte;
119   return 0;
120 }
121 
i2cdetect_dash_F(int bus)122 static void i2cdetect_dash_F(int bus)
123 {
124   size_t i;
125 
126   struct { int mask; const char *name; } funcs[] = {
127     {I2C_FUNC_I2C, "I2C"},
128     {I2C_FUNC_SMBUS_QUICK, "SMBus Quick Command"},
129     {I2C_FUNC_SMBUS_WRITE_BYTE, "SMBus Send Byte"},
130     {I2C_FUNC_SMBUS_READ_BYTE, "SMBus Receive Byte"},
131     {I2C_FUNC_SMBUS_WRITE_BYTE_DATA, "SMBus Write Byte"},
132     {I2C_FUNC_SMBUS_READ_BYTE_DATA, "SMBus Read Byte"},
133     {I2C_FUNC_SMBUS_WRITE_WORD_DATA, "SMBus Write Word"},
134     {I2C_FUNC_SMBUS_READ_WORD_DATA, "SMBus Read Word"},
135     {I2C_FUNC_SMBUS_PROC_CALL, "SMBus Process Call"},
136     {I2C_FUNC_SMBUS_WRITE_BLOCK_DATA, "SMBus Write Block"},
137     {I2C_FUNC_SMBUS_READ_BLOCK_DATA, "SMBus Read Block"},
138     {I2C_FUNC_SMBUS_BLOCK_PROC_CALL, "SMBus Block Process Call"},
139     {I2C_FUNC_SMBUS_PEC, "SMBus PEC"},
140     {I2C_FUNC_SMBUS_WRITE_I2C_BLOCK, "I2C Write Block"},
141     {I2C_FUNC_SMBUS_READ_I2C_BLOCK, "I2C Read Block"},
142   };
143   unsigned long supported = i2c_get_funcs(bus);
144 
145   printf("Functionalities implemented by %s:\n", toybuf);
146   for (i = 0; i < ARRAY_LEN(funcs); ++i) {
147     printf("%-32s %s\n", funcs[i].name,
148            (supported & funcs[i].mask) ? "yes" : "no");
149   }
150 }
151 
i2cdetect_dash_l(struct dirtree * node)152 static int i2cdetect_dash_l(struct dirtree *node)
153 {
154   int suffix_len = strlen("/name");
155   int bus;
156   char *fname, *p;
157   unsigned long funcs;
158 
159   if (!node->parent) return DIRTREE_RECURSE; // Skip the directory itself.
160 
161   if (sscanf(node->name, "i2c-%d", &bus) != 1) return 0;
162   funcs = i2c_get_funcs(bus);
163 
164   fname = dirtree_path(node, &suffix_len);
165   strcat(fname, "/name");
166   xreadfile(fname, toybuf, sizeof(toybuf));
167   free(fname);
168   if ((p = strchr(toybuf, '\n'))) *p = 0;
169 
170   // "i2c-1	i2c       	Synopsys DesignWare I2C adapter 	I2C adapter"
171   printf("%s\t%-10s\t%-32s\t%s\n", node->name,
172          (funcs & I2C_FUNC_I2C) ? "i2c" : "?",
173          toybuf,
174          (funcs & I2C_FUNC_I2C) ? "I2C Adapter" : "?");
175 
176   return 0;
177 }
178 
i2cdetect_main(void)179 void i2cdetect_main(void)
180 {
181   if (toys.optflags & FLAG_l) {
182     if (toys.optc) error_exit("-l doesn't take arguments");
183     dirtree_read("/sys/class/i2c-dev", i2cdetect_dash_l);
184   } else if (toys.optflags & FLAG_F) {
185     if (toys.optc != 1) error_exit("-F BUS");
186     i2cdetect_dash_F(atolx_range(*toys.optargs, 0, INT_MAX));
187   } else {
188     int bus, first = 0x03, last = 0x77, fd, row, addr, byte;
189 
190     if (toys.optflags & FLAG_a) {
191       first = 0x00;
192       last = 0x7f;
193     }
194 
195     if (toys.optc != 1 && toys.optc != 3) error_exit("bad args");
196     bus = atolx_range(*toys.optargs, 0, INT_MAX);
197     if (toys.optc == 3) {
198       first = atolx_range(toys.optargs[1], 0, 0x7f);
199       last = atolx_range(toys.optargs[2], 0, 0x7f);
200       if (first > last) error_exit("first > last");
201     }
202 
203     confirm("Probe chips 0x%02x-0x%02x on bus %d?", first, last, bus);
204 
205     fd = i2c_open(bus, 0, 0);
206     printf("     0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f\n");
207     for (row = 0; row <= 0x70; row += 16) {
208       xprintf("%02x:", row & 0xf0);
209       for (addr = row; addr < row + 16; ++addr) {
210         if (addr < first || addr > last) printf("   ");
211         else {
212           if (ioctl(fd, I2C_SLAVE, addr) == -1) {
213             if (errno == EBUSY) {
214               xprintf(" UU");
215               continue;
216             }
217             perror_exit("ioctl(I2C_SLAVE)");
218           }
219           if (i2c_read_byte(fd, addr, &byte) == -1) xprintf(" --");
220           else xprintf(" %02x", addr);
221         }
222       }
223       putchar('\n');
224     }
225     close(fd);
226   }
227 }
228 
229 #define CLEANUP_i2cdetect
230 #define FOR_i2cdump
231 #include "generated/flags.h"
232 
i2cdump_main(void)233 void i2cdump_main(void)
234 {
235   int bus = atolx_range(toys.optargs[0], 0, INT_MAX);
236   int chip = atolx_range(toys.optargs[1], 0, 0x7f);
237   int fd, row, addr, byte;
238 
239   confirm("Dump chip 0x%02x on bus %d?", chip, bus);
240 
241   fd = i2c_open(bus, (toys.optflags&FLAG_f)?I2C_SLAVE_FORCE:I2C_SLAVE, chip);
242   printf("     0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f    0123456789abcdef\n");
243   for (row = 0; row <= 0xf0; row += 16) {
244     xprintf("%02x:", row & 0xf0);
245     for (addr = row; addr < row + 16; ++addr) {
246       if (i2c_read_byte(fd, addr, &byte) == -1) perror_exit("i2c_read_byte");
247       printf(" %02x", byte);
248       toybuf[addr-row] = isprint(byte) ? byte : (byte ? '?' : '.');
249     }
250     printf("    %16.16s\n", toybuf);
251   }
252   close(fd);
253 }
254 
255 #define CLEANUP_i2cdump
256 #define FOR_i2cget
257 #include "generated/flags.h"
258 
i2cget_main(void)259 void i2cget_main(void)
260 {
261   int bus = atolx_range(toys.optargs[0], 0, INT_MAX);
262   int chip = atolx_range(toys.optargs[1], 0, 0x7f);
263   int addr = atolx_range(toys.optargs[2], 0, 0xff);
264   int fd, byte;
265 
266   confirm("Read register 0x%02x from chip 0x%02x on bus %d?", addr, chip, bus);
267 
268   fd = i2c_open(bus, (toys.optflags&FLAG_f)?I2C_SLAVE_FORCE:I2C_SLAVE, chip);
269   if (i2c_read_byte(fd, addr, &byte) == -1) perror_exit("i2c_read_byte");
270   printf("0x%02x\n", byte);
271   close(fd);
272 }
273 
274 #define CLEANUP_i2cget
275 #define FOR_i2cset
276 #include "generated/flags.h"
277 
i2cset_main(void)278 void i2cset_main(void)
279 {
280   int bus = atolx_range(toys.optargs[0], 0, INT_MAX);
281   int chip = atolx_range(toys.optargs[1], 0, 0x7f);
282   int addr = atolx_range(toys.optargs[2], 0, 0xff);
283   char *mode = toys.optargs[toys.optc-1];
284   int fd, i;
285   struct i2c_smbus_ioctl_data ioctl_data;
286   union i2c_smbus_data data;
287 
288   memset(&data, 0, sizeof(data));
289   if (strlen(mode) != 1) help_exit("mode too long");
290   if (*mode == 'b' && toys.optc == 5) {
291     ioctl_data.size = I2C_SMBUS_BYTE_DATA;
292     data.byte = atolx_range(toys.optargs[3], 0, 0xff);
293   } else if (*mode == 'w' && toys.optc == 5) {
294     ioctl_data.size = I2C_SMBUS_WORD_DATA;
295     data.word = atolx_range(toys.optargs[3], 0, 0xffff);
296   } else if (*mode == 'i' && toys.optc >= 5) {
297     if (toys.optc - 4 > I2C_SMBUS_BLOCK_MAX) error_exit("too much data");
298     ioctl_data.size = I2C_SMBUS_I2C_BLOCK_DATA;
299     for (i = 0; i < toys.optc - 4; ++i)
300       data.block[i+1] = atolx_range(toys.optargs[3+i], 0, 0xff);
301     data.block[0] = toys.optc - 4;
302   } else {
303     help_exit("syntax error");
304   }
305 
306   confirm("Write register 0x%02x from chip 0x%02x on bus %d?", addr, chip, bus);
307 
308   fd = i2c_open(bus, (toys.optflags&FLAG_f)?I2C_SLAVE_FORCE:I2C_SLAVE, chip);
309   ioctl_data.read_write = I2C_SMBUS_WRITE;
310   ioctl_data.command = addr;
311   ioctl_data.data = &data;
312   xioctl(fd, I2C_SMBUS, &ioctl_data);
313   close(fd);
314 }
315