blob: 9ebbdd312840242728607d8e1a705d9a2f61af27 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenk81a88242002-10-26 15:22:42 +00002/*
Heiko Schochere0e55bc2012-01-16 21:12:24 +00003 * (C) Copyright 2009
4 * Sergey Kubushyn, himself, ksi@koi8.net
5 *
6 * Changes for unified multibus/multiadapter I2C support.
7 *
wdenk81a88242002-10-26 15:22:42 +00008 * (C) Copyright 2001
9 * Gerald Van Baren, Custom IDEAS, vanbaren@cideas.com.
wdenk81a88242002-10-26 15:22:42 +000010 */
11
12/*
13 * I2C Functions similar to the standard memory functions.
14 *
15 * There are several parameters in many of the commands that bear further
16 * explanations:
17 *
wdenk81a88242002-10-26 15:22:42 +000018 * {i2c_chip} is the I2C chip address (the first byte sent on the bus).
19 * Each I2C chip on the bus has a unique address. On the I2C data bus,
20 * the address is the upper seven bits and the LSB is the "read/write"
21 * bit. Note that the {i2c_chip} address specified on the command
22 * line is not shifted up: e.g. a typical EEPROM memory chip may have
23 * an I2C address of 0x50, but the data put on the bus will be 0xA0
24 * for write and 0xA1 for read. This "non shifted" address notation
25 * matches at least half of the data sheets :-/.
26 *
27 * {addr} is the address (or offset) within the chip. Small memory
28 * chips have 8 bit addresses. Large memory chips have 16 bit
29 * addresses. Other memory chips have 9, 10, or 11 bit addresses.
30 * Many non-memory chips have multiple registers and {addr} is used
31 * as the register index. Some non-memory chips have only one register
32 * and therefore don't need any {addr} parameter.
33 *
34 * The default {addr} parameter is one byte (.1) which works well for
35 * memories and registers with 8 bits of address space.
36 *
37 * You can specify the length of the {addr} field with the optional .0,
38 * .1, or .2 modifier (similar to the .b, .w, .l modifier). If you are
39 * manipulating a single register device which doesn't use an address
40 * field, use "0.0" for the address and the ".0" length field will
41 * suppress the address in the I2C data stream. This also works for
42 * successive reads using the I2C auto-incrementing memory pointer.
43 *
44 * If you are manipulating a large memory with 2-byte addresses, use
45 * the .2 address modifier, e.g. 210.2 addresses location 528 (decimal).
46 *
47 * Then there are the unfortunate memory chips that spill the most
48 * significant 1, 2, or 3 bits of address into the chip address byte.
49 * This effectively makes one chip (logically) look like 2, 4, or
50 * 8 chips. This is handled (awkwardly) by #defining
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +020051 * CONFIG_SYS_I2C_EEPROM_ADDR_OVERFLOW and using the .1 modifier on the
wdenk81a88242002-10-26 15:22:42 +000052 * {addr} field (since .1 is the default, it doesn't actually have to
53 * be specified). Examples: given a memory chip at I2C chip address
54 * 0x50, the following would happen...
Peter Tyser469cde42009-04-18 22:34:03 -050055 * i2c md 50 0 10 display 16 bytes starting at 0x000
wdenk81a88242002-10-26 15:22:42 +000056 * On the bus: <S> A0 00 <E> <S> A1 <rd> ... <rd>
Peter Tyser469cde42009-04-18 22:34:03 -050057 * i2c md 50 100 10 display 16 bytes starting at 0x100
wdenk81a88242002-10-26 15:22:42 +000058 * On the bus: <S> A2 00 <E> <S> A3 <rd> ... <rd>
Peter Tyser469cde42009-04-18 22:34:03 -050059 * i2c md 50 210 10 display 16 bytes starting at 0x210
wdenk81a88242002-10-26 15:22:42 +000060 * On the bus: <S> A4 10 <E> <S> A5 <rd> ... <rd>
61 * This is awfully ugly. It would be nice if someone would think up
62 * a better way of handling this.
63 *
64 * Adapted from cmd_mem.c which is copyright Wolfgang Denk (wd@denx.de).
65 */
66
67#include <common.h>
Simon Glass399ed9a2014-04-10 20:01:30 -060068#include <bootretry.h>
Simon Glassdec3c012014-04-10 20:01:25 -060069#include <cli.h>
wdenk81a88242002-10-26 15:22:42 +000070#include <command.h>
Simon Glassa73bda42015-11-08 23:47:45 -070071#include <console.h>
Simon Glassc11ddd42014-12-10 08:55:48 -070072#include <dm.h>
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +000073#include <edid.h>
Simon Glassc11ddd42014-12-10 08:55:48 -070074#include <errno.h>
wdenk81a88242002-10-26 15:22:42 +000075#include <i2c.h>
Heiko Schocher6ee861b2008-10-15 09:39:47 +020076#include <malloc.h>
wdenk81a88242002-10-26 15:22:42 +000077#include <asm/byteorder.h>
Marek Vasut392d9242012-11-12 14:34:25 +000078#include <linux/compiler.h>
Simon Glass48b6c6b2019-11-14 12:57:16 -070079#include <u-boot/crc.h>
wdenk81a88242002-10-26 15:22:42 +000080
wdenk81a88242002-10-26 15:22:42 +000081/* Display values from last command.
82 * Memory modify remembered values are different from display memory.
83 */
Masahiro Yamadab87abaf2014-12-20 03:34:23 +090084static uint i2c_dp_last_chip;
wdenk81a88242002-10-26 15:22:42 +000085static uint i2c_dp_last_addr;
86static uint i2c_dp_last_alen;
87static uint i2c_dp_last_length = 0x10;
88
Masahiro Yamadab87abaf2014-12-20 03:34:23 +090089static uint i2c_mm_last_chip;
wdenk81a88242002-10-26 15:22:42 +000090static uint i2c_mm_last_addr;
91static uint i2c_mm_last_alen;
92
Ben Warren45657152006-09-07 16:50:54 -040093/* If only one I2C bus is present, the list of devices to ignore when
94 * the probe command is issued is represented by a 1D array of addresses.
95 * When multiple buses are present, the list is an array of bus-address
96 * pairs. The following macros take care of this */
97
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +020098#if defined(CONFIG_SYS_I2C_NOPROBES)
Heiko Schochera4ea4452012-10-25 10:32:14 +020099#if defined(CONFIG_SYS_I2C) || defined(CONFIG_I2C_MULTI_BUS)
Ben Warren45657152006-09-07 16:50:54 -0400100static struct
101{
102 uchar bus;
103 uchar addr;
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200104} i2c_no_probes[] = CONFIG_SYS_I2C_NOPROBES;
Ben Warren45657152006-09-07 16:50:54 -0400105#define GET_BUS_NUM i2c_get_bus_num()
106#define COMPARE_BUS(b,i) (i2c_no_probes[(i)].bus == (b))
107#define COMPARE_ADDR(a,i) (i2c_no_probes[(i)].addr == (a))
108#define NO_PROBE_ADDR(i) i2c_no_probes[(i)].addr
109#else /* single bus */
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200110static uchar i2c_no_probes[] = CONFIG_SYS_I2C_NOPROBES;
Ben Warren45657152006-09-07 16:50:54 -0400111#define GET_BUS_NUM 0
112#define COMPARE_BUS(b,i) ((b) == 0) /* Make compiler happy */
113#define COMPARE_ADDR(a,i) (i2c_no_probes[(i)] == (a))
114#define NO_PROBE_ADDR(i) i2c_no_probes[(i)]
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000115#endif /* defined(CONFIG_SYS_I2C) */
Heiko Schocher6ee861b2008-10-15 09:39:47 +0200116#endif
117
Frans Meulenbroeks97ea6e92010-03-26 09:46:40 +0100118#define DISP_LINE_LEN 16
119
Simon Glassc11ddd42014-12-10 08:55:48 -0700120/*
121 * Default for driver model is to use the chip's existing address length.
122 * For legacy code, this is not stored, so we need to use a suitable
123 * default.
124 */
125#ifdef CONFIG_DM_I2C
126#define DEFAULT_ADDR_LEN (-1)
127#else
128#define DEFAULT_ADDR_LEN 1
129#endif
130
131#ifdef CONFIG_DM_I2C
132static struct udevice *i2c_cur_bus;
133
Simon Glass7d722762015-01-12 18:02:07 -0700134static int cmd_i2c_set_bus_num(unsigned int busnum)
Simon Glassc11ddd42014-12-10 08:55:48 -0700135{
136 struct udevice *bus;
137 int ret;
138
139 ret = uclass_get_device_by_seq(UCLASS_I2C, busnum, &bus);
140 if (ret) {
141 debug("%s: No bus %d\n", __func__, busnum);
142 return ret;
143 }
144 i2c_cur_bus = bus;
145
146 return 0;
147}
148
149static int i2c_get_cur_bus(struct udevice **busp)
150{
Lukasz Majewski0a556272017-03-21 12:08:25 +0100151#ifdef CONFIG_I2C_SET_DEFAULT_BUS_NUM
152 if (!i2c_cur_bus) {
153 if (cmd_i2c_set_bus_num(CONFIG_I2C_DEFAULT_BUS_NUMBER)) {
154 printf("Default I2C bus %d not found\n",
155 CONFIG_I2C_DEFAULT_BUS_NUMBER);
156 return -ENODEV;
157 }
158 }
159#endif
160
Simon Glassc11ddd42014-12-10 08:55:48 -0700161 if (!i2c_cur_bus) {
162 puts("No I2C bus selected\n");
163 return -ENODEV;
164 }
165 *busp = i2c_cur_bus;
166
167 return 0;
168}
169
170static int i2c_get_cur_bus_chip(uint chip_addr, struct udevice **devp)
171{
172 struct udevice *bus;
173 int ret;
174
175 ret = i2c_get_cur_bus(&bus);
176 if (ret)
177 return ret;
178
Simon Glassa2723ae2015-01-25 08:26:55 -0700179 return i2c_get_chip(bus, chip_addr, 1, devp);
Simon Glassc11ddd42014-12-10 08:55:48 -0700180}
181
182#endif
183
Marek Vasut2476c8d2012-11-12 14:34:27 +0000184/**
185 * i2c_init_board() - Board-specific I2C bus init
186 *
187 * This function is the default no-op implementation of I2C bus
Robert P. J. Day8d56db92016-07-15 13:44:45 -0400188 * initialization. This function can be overridden by board-specific
Marek Vasut2476c8d2012-11-12 14:34:27 +0000189 * implementation if needed.
190 */
Marek Vasut392d9242012-11-12 14:34:25 +0000191__weak
192void i2c_init_board(void)
Stefan Bigler2f8baaa2011-04-08 16:24:08 +0200193{
Stefan Bigler2f8baaa2011-04-08 16:24:08 +0200194}
Stefan Bigler2f8baaa2011-04-08 16:24:08 +0200195
Peter Tyser2e697652009-04-18 22:34:01 -0500196/* TODO: Implement architecture-specific get/set functions */
Marek Vasut2476c8d2012-11-12 14:34:27 +0000197
198/**
199 * i2c_get_bus_speed() - Return I2C bus speed
200 *
201 * This function is the default implementation of function for retrieveing
202 * the current I2C bus speed in Hz.
203 *
204 * A driver implementing runtime switching of I2C bus speed must override
205 * this function to report the speed correctly. Simple or legacy drivers
206 * can use this fallback.
207 *
208 * Returns I2C bus speed in Hz.
209 */
Simon Glassc11ddd42014-12-10 08:55:48 -0700210#if !defined(CONFIG_SYS_I2C) && !defined(CONFIG_DM_I2C)
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000211/*
212 * TODO: Implement architecture-specific get/set functions
213 * Should go away, if we switched completely to new multibus support
214 */
Marek Vasut392d9242012-11-12 14:34:25 +0000215__weak
216unsigned int i2c_get_bus_speed(void)
Peter Tyser2e697652009-04-18 22:34:01 -0500217{
218 return CONFIG_SYS_I2C_SPEED;
219}
Peter Tyser2e697652009-04-18 22:34:01 -0500220
Marek Vasut2476c8d2012-11-12 14:34:27 +0000221/**
222 * i2c_set_bus_speed() - Configure I2C bus speed
223 * @speed: Newly set speed of the I2C bus in Hz
224 *
225 * This function is the default implementation of function for setting
226 * the I2C bus speed in Hz.
227 *
228 * A driver implementing runtime switching of I2C bus speed must override
229 * this function to report the speed correctly. Simple or legacy drivers
230 * can use this fallback.
231 *
232 * Returns zero on success, negative value on error.
233 */
Marek Vasut392d9242012-11-12 14:34:25 +0000234__weak
235int i2c_set_bus_speed(unsigned int speed)
Peter Tyser2e697652009-04-18 22:34:01 -0500236{
237 if (speed != CONFIG_SYS_I2C_SPEED)
238 return -1;
239
240 return 0;
241}
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000242#endif
Peter Tyser2e697652009-04-18 22:34:01 -0500243
Marek Vasut2476c8d2012-11-12 14:34:27 +0000244/**
245 * get_alen() - Small parser helper function to get address length
246 *
247 * Returns the address length.
Frans Meulenbroeksd388e882010-03-26 09:46:41 +0100248 */
Simon Glassc11ddd42014-12-10 08:55:48 -0700249static uint get_alen(char *arg, int default_len)
Frans Meulenbroeksd388e882010-03-26 09:46:41 +0100250{
251 int j;
252 int alen;
253
Simon Glassc11ddd42014-12-10 08:55:48 -0700254 alen = default_len;
Frans Meulenbroeksd388e882010-03-26 09:46:41 +0100255 for (j = 0; j < 8; j++) {
256 if (arg[j] == '.') {
257 alen = arg[j+1] - '0';
Frans Meulenbroeksd388e882010-03-26 09:46:41 +0100258 break;
259 } else if (arg[j] == '\0')
260 break;
261 }
262 return alen;
263}
264
Simon Glass2a302392014-11-11 10:46:17 -0700265enum i2c_err_op {
266 I2C_ERR_READ,
267 I2C_ERR_WRITE,
268};
269
270static int i2c_report_err(int ret, enum i2c_err_op op)
271{
272 printf("Error %s the chip: %d\n",
273 op == I2C_ERR_READ ? "reading" : "writing", ret);
274
275 return CMD_RET_FAILURE;
276}
277
Marek Vasut2476c8d2012-11-12 14:34:27 +0000278/**
279 * do_i2c_read() - Handle the "i2c read" command-line command
280 * @cmdtp: Command data struct pointer
281 * @flag: Command flag
282 * @argc: Command-line argument count
283 * @argv: Array of command-line arguments
284 *
285 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
286 * on error.
287 *
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100288 * Syntax:
289 * i2c read {i2c_chip} {devaddr}{.0, .1, .2} {len} {memaddr}
290 */
Simon Glassed38aef2020-05-10 11:40:03 -0600291static int do_i2c_read(struct cmd_tbl *cmdtp, int flag, int argc,
292 char *const argv[])
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100293{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900294 uint chip;
Simon Glassc11ddd42014-12-10 08:55:48 -0700295 uint devaddr, length;
296 int alen;
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100297 u_char *memaddr;
Simon Glassc11ddd42014-12-10 08:55:48 -0700298 int ret;
299#ifdef CONFIG_DM_I2C
300 struct udevice *dev;
301#endif
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100302
Wolfgang Denk3b683112010-07-17 01:06:04 +0200303 if (argc != 5)
Simon Glassa06dfc72011-12-10 08:44:01 +0000304 return CMD_RET_USAGE;
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100305
306 /*
307 * I2C chip address
308 */
309 chip = simple_strtoul(argv[1], NULL, 16);
310
311 /*
312 * I2C data address within the chip. This can be 1 or
313 * 2 bytes long. Some day it might be 3 bytes long :-).
314 */
315 devaddr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700316 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +0200317 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000318 return CMD_RET_USAGE;
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100319
320 /*
321 * Length is the number of objects, not number of bytes.
322 */
323 length = simple_strtoul(argv[3], NULL, 16);
324
325 /*
326 * memaddr is the address where to store things in memory
327 */
328 memaddr = (u_char *)simple_strtoul(argv[4], NULL, 16);
329
Simon Glassc11ddd42014-12-10 08:55:48 -0700330#ifdef CONFIG_DM_I2C
331 ret = i2c_get_cur_bus_chip(chip, &dev);
332 if (!ret && alen != -1)
333 ret = i2c_set_chip_offset_len(dev, alen);
334 if (!ret)
Simon Glass7d722762015-01-12 18:02:07 -0700335 ret = dm_i2c_read(dev, devaddr, memaddr, length);
Simon Glassc11ddd42014-12-10 08:55:48 -0700336#else
337 ret = i2c_read(chip, devaddr, alen, memaddr, length);
338#endif
339 if (ret)
340 return i2c_report_err(ret, I2C_ERR_READ);
341
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +0100342 return 0;
343}
344
Simon Glassed38aef2020-05-10 11:40:03 -0600345static int do_i2c_write(struct cmd_tbl *cmdtp, int flag, int argc,
346 char *const argv[])
York Sun53bb44d2012-09-16 08:02:30 +0000347{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900348 uint chip;
Simon Glassc11ddd42014-12-10 08:55:48 -0700349 uint devaddr, length;
350 int alen;
York Sun53bb44d2012-09-16 08:02:30 +0000351 u_char *memaddr;
Simon Glassc11ddd42014-12-10 08:55:48 -0700352 int ret;
353#ifdef CONFIG_DM_I2C
354 struct udevice *dev;
Lubomir Popov56dae702015-01-30 19:56:04 +0200355 struct dm_i2c_chip *i2c_chip;
Simon Glassc11ddd42014-12-10 08:55:48 -0700356#endif
York Sun53bb44d2012-09-16 08:02:30 +0000357
Lubomir Popov56dae702015-01-30 19:56:04 +0200358 if ((argc < 5) || (argc > 6))
York Sun53bb44d2012-09-16 08:02:30 +0000359 return cmd_usage(cmdtp);
360
361 /*
362 * memaddr is the address where to store things in memory
363 */
364 memaddr = (u_char *)simple_strtoul(argv[1], NULL, 16);
365
366 /*
367 * I2C chip address
368 */
369 chip = simple_strtoul(argv[2], NULL, 16);
370
371 /*
372 * I2C data address within the chip. This can be 1 or
373 * 2 bytes long. Some day it might be 3 bytes long :-).
374 */
375 devaddr = simple_strtoul(argv[3], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700376 alen = get_alen(argv[3], DEFAULT_ADDR_LEN);
York Sun53bb44d2012-09-16 08:02:30 +0000377 if (alen > 3)
378 return cmd_usage(cmdtp);
379
380 /*
Lubomir Popov56dae702015-01-30 19:56:04 +0200381 * Length is the number of bytes.
York Sun53bb44d2012-09-16 08:02:30 +0000382 */
383 length = simple_strtoul(argv[4], NULL, 16);
384
Simon Glassc11ddd42014-12-10 08:55:48 -0700385#ifdef CONFIG_DM_I2C
386 ret = i2c_get_cur_bus_chip(chip, &dev);
387 if (!ret && alen != -1)
388 ret = i2c_set_chip_offset_len(dev, alen);
389 if (ret)
390 return i2c_report_err(ret, I2C_ERR_WRITE);
Lubomir Popov56dae702015-01-30 19:56:04 +0200391 i2c_chip = dev_get_parent_platdata(dev);
392 if (!i2c_chip)
393 return i2c_report_err(ret, I2C_ERR_WRITE);
Simon Glassc11ddd42014-12-10 08:55:48 -0700394#endif
395
Lubomir Popov56dae702015-01-30 19:56:04 +0200396 if (argc == 6 && !strcmp(argv[5], "-s")) {
397 /*
398 * Write all bytes in a single I2C transaction. If the target
399 * device is an EEPROM, it is your responsibility to not cross
400 * a page boundary. No write delay upon completion, take this
401 * into account if linking commands.
402 */
Simon Glassc11ddd42014-12-10 08:55:48 -0700403#ifdef CONFIG_DM_I2C
Lubomir Popov56dae702015-01-30 19:56:04 +0200404 i2c_chip->flags &= ~DM_I2C_CHIP_WR_ADDRESS;
405 ret = dm_i2c_write(dev, devaddr, memaddr, length);
Simon Glassc11ddd42014-12-10 08:55:48 -0700406#else
Lubomir Popov56dae702015-01-30 19:56:04 +0200407 ret = i2c_write(chip, devaddr, alen, memaddr, length);
Simon Glassc11ddd42014-12-10 08:55:48 -0700408#endif
409 if (ret)
410 return i2c_report_err(ret, I2C_ERR_WRITE);
Lubomir Popov56dae702015-01-30 19:56:04 +0200411 } else {
412 /*
413 * Repeated addressing - perform <length> separate
414 * write transactions of one byte each
415 */
416 while (length-- > 0) {
417#ifdef CONFIG_DM_I2C
418 i2c_chip->flags |= DM_I2C_CHIP_WR_ADDRESS;
419 ret = dm_i2c_write(dev, devaddr++, memaddr++, 1);
420#else
421 ret = i2c_write(chip, devaddr++, alen, memaddr++, 1);
422#endif
423 if (ret)
424 return i2c_report_err(ret, I2C_ERR_WRITE);
York Sun53bb44d2012-09-16 08:02:30 +0000425/*
426 * No write delay with FRAM devices.
427 */
428#if !defined(CONFIG_SYS_I2C_FRAM)
Lubomir Popov56dae702015-01-30 19:56:04 +0200429 udelay(11000);
York Sun53bb44d2012-09-16 08:02:30 +0000430#endif
Lubomir Popov56dae702015-01-30 19:56:04 +0200431 }
York Sun53bb44d2012-09-16 08:02:30 +0000432 }
Simon Glassc11ddd42014-12-10 08:55:48 -0700433 return 0;
434}
435
436#ifdef CONFIG_DM_I2C
Simon Glassed38aef2020-05-10 11:40:03 -0600437static int do_i2c_flags(struct cmd_tbl *cmdtp, int flag, int argc,
Simon Glassc11ddd42014-12-10 08:55:48 -0700438 char *const argv[])
439{
440 struct udevice *dev;
441 uint flags;
442 int chip;
443 int ret;
444
445 if (argc < 2)
446 return CMD_RET_USAGE;
447
448 chip = simple_strtoul(argv[1], NULL, 16);
449 ret = i2c_get_cur_bus_chip(chip, &dev);
450 if (ret)
451 return i2c_report_err(ret, I2C_ERR_READ);
452
453 if (argc > 2) {
454 flags = simple_strtoul(argv[2], NULL, 16);
455 ret = i2c_set_chip_flags(dev, flags);
456 } else {
457 ret = i2c_get_chip_flags(dev, &flags);
458 if (!ret)
459 printf("%x\n", flags);
460 }
461 if (ret)
462 return i2c_report_err(ret, I2C_ERR_READ);
463
York Sun53bb44d2012-09-16 08:02:30 +0000464 return 0;
465}
Simon Glassa44d5df2015-08-22 18:31:33 -0600466
Simon Glassed38aef2020-05-10 11:40:03 -0600467static int do_i2c_olen(struct cmd_tbl *cmdtp, int flag, int argc,
468 char *const argv[])
Simon Glassa44d5df2015-08-22 18:31:33 -0600469{
470 struct udevice *dev;
471 uint olen;
472 int chip;
473 int ret;
474
475 if (argc < 2)
476 return CMD_RET_USAGE;
477
478 chip = simple_strtoul(argv[1], NULL, 16);
479 ret = i2c_get_cur_bus_chip(chip, &dev);
480 if (ret)
481 return i2c_report_err(ret, I2C_ERR_READ);
482
483 if (argc > 2) {
484 olen = simple_strtoul(argv[2], NULL, 16);
485 ret = i2c_set_chip_offset_len(dev, olen);
486 } else {
487 ret = i2c_get_chip_offset_len(dev);
488 if (ret >= 0) {
489 printf("%x\n", ret);
490 ret = 0;
491 }
492 }
493 if (ret)
494 return i2c_report_err(ret, I2C_ERR_READ);
495
496 return 0;
497}
Simon Glassc11ddd42014-12-10 08:55:48 -0700498#endif
York Sun53bb44d2012-09-16 08:02:30 +0000499
Marek Vasut2476c8d2012-11-12 14:34:27 +0000500/**
501 * do_i2c_md() - Handle the "i2c md" command-line command
502 * @cmdtp: Command data struct pointer
503 * @flag: Command flag
504 * @argc: Command-line argument count
505 * @argv: Array of command-line arguments
506 *
507 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
508 * on error.
509 *
Frans Meulenbroeks6a08fd12010-03-26 09:46:39 +0100510 * Syntax:
511 * i2c md {i2c_chip} {addr}{.0, .1, .2} {len}
512 */
Simon Glassed38aef2020-05-10 11:40:03 -0600513static int do_i2c_md(struct cmd_tbl *cmdtp, int flag, int argc,
514 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +0000515{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900516 uint chip;
Simon Glassc11ddd42014-12-10 08:55:48 -0700517 uint addr, length;
518 int alen;
wdenk81a88242002-10-26 15:22:42 +0000519 int j, nbytes, linebytes;
Simon Glassc11ddd42014-12-10 08:55:48 -0700520 int ret;
521#ifdef CONFIG_DM_I2C
522 struct udevice *dev;
523#endif
wdenk81a88242002-10-26 15:22:42 +0000524
525 /* We use the last specified parameters, unless new ones are
526 * entered.
527 */
528 chip = i2c_dp_last_chip;
529 addr = i2c_dp_last_addr;
530 alen = i2c_dp_last_alen;
531 length = i2c_dp_last_length;
532
Wolfgang Denk3b683112010-07-17 01:06:04 +0200533 if (argc < 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000534 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000535
536 if ((flag & CMD_FLAG_REPEAT) == 0) {
537 /*
538 * New command specified.
539 */
wdenk81a88242002-10-26 15:22:42 +0000540
541 /*
542 * I2C chip address
543 */
544 chip = simple_strtoul(argv[1], NULL, 16);
545
546 /*
547 * I2C data address within the chip. This can be 1 or
548 * 2 bytes long. Some day it might be 3 bytes long :-).
549 */
550 addr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700551 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +0200552 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000553 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000554
555 /*
556 * If another parameter, it is the length to display.
557 * Length is the number of objects, not number of bytes.
558 */
559 if (argc > 3)
560 length = simple_strtoul(argv[3], NULL, 16);
561 }
562
Simon Glassc11ddd42014-12-10 08:55:48 -0700563#ifdef CONFIG_DM_I2C
564 ret = i2c_get_cur_bus_chip(chip, &dev);
565 if (!ret && alen != -1)
566 ret = i2c_set_chip_offset_len(dev, alen);
567 if (ret)
568 return i2c_report_err(ret, I2C_ERR_READ);
569#endif
570
wdenk81a88242002-10-26 15:22:42 +0000571 /*
572 * Print the lines.
573 *
574 * We buffer all read data, so we can make sure data is read only
575 * once.
576 */
577 nbytes = length;
578 do {
579 unsigned char linebuf[DISP_LINE_LEN];
580 unsigned char *cp;
581
582 linebytes = (nbytes > DISP_LINE_LEN) ? DISP_LINE_LEN : nbytes;
583
Simon Glassc11ddd42014-12-10 08:55:48 -0700584#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700585 ret = dm_i2c_read(dev, addr, linebuf, linebytes);
Simon Glassc11ddd42014-12-10 08:55:48 -0700586#else
587 ret = i2c_read(chip, addr, alen, linebuf, linebytes);
588#endif
589 if (ret)
Masahiro Yamada6b546f32015-02-05 13:50:26 +0900590 return i2c_report_err(ret, I2C_ERR_READ);
Timur Tabiff0215a2006-11-28 12:09:35 -0600591 else {
wdenk81a88242002-10-26 15:22:42 +0000592 printf("%04x:", addr);
593 cp = linebuf;
594 for (j=0; j<linebytes; j++) {
595 printf(" %02x", *cp++);
596 addr++;
597 }
wdenk42c05472004-03-23 22:14:11 +0000598 puts (" ");
wdenk81a88242002-10-26 15:22:42 +0000599 cp = linebuf;
600 for (j=0; j<linebytes; j++) {
601 if ((*cp < 0x20) || (*cp > 0x7e))
wdenk42c05472004-03-23 22:14:11 +0000602 puts (".");
wdenk81a88242002-10-26 15:22:42 +0000603 else
604 printf("%c", *cp);
605 cp++;
606 }
wdenk42c05472004-03-23 22:14:11 +0000607 putc ('\n');
wdenk81a88242002-10-26 15:22:42 +0000608 }
609 nbytes -= linebytes;
610 } while (nbytes > 0);
611
612 i2c_dp_last_chip = chip;
613 i2c_dp_last_addr = addr;
614 i2c_dp_last_alen = alen;
615 i2c_dp_last_length = length;
616
617 return 0;
618}
619
Marek Vasut2476c8d2012-11-12 14:34:27 +0000620/**
621 * do_i2c_mw() - Handle the "i2c mw" command-line command
622 * @cmdtp: Command data struct pointer
623 * @flag: Command flag
624 * @argc: Command-line argument count
625 * @argv: Array of command-line arguments
626 *
627 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
628 * on error.
wdenk81a88242002-10-26 15:22:42 +0000629 *
630 * Syntax:
Peter Tyser469cde42009-04-18 22:34:03 -0500631 * i2c mw {i2c_chip} {addr}{.0, .1, .2} {data} [{count}]
wdenk81a88242002-10-26 15:22:42 +0000632 */
Simon Glassed38aef2020-05-10 11:40:03 -0600633static int do_i2c_mw(struct cmd_tbl *cmdtp, int flag, int argc,
634 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +0000635{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900636 uint chip;
wdenk81a88242002-10-26 15:22:42 +0000637 ulong addr;
Simon Glassc11ddd42014-12-10 08:55:48 -0700638 int alen;
wdenk81a88242002-10-26 15:22:42 +0000639 uchar byte;
640 int count;
Simon Glassc11ddd42014-12-10 08:55:48 -0700641 int ret;
642#ifdef CONFIG_DM_I2C
643 struct udevice *dev;
644#endif
wdenk81a88242002-10-26 15:22:42 +0000645
Wolfgang Denk3b683112010-07-17 01:06:04 +0200646 if ((argc < 4) || (argc > 5))
Simon Glassa06dfc72011-12-10 08:44:01 +0000647 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000648
649 /*
Wolfgang Denka1be4762008-05-20 16:00:29 +0200650 * Chip is always specified.
651 */
wdenk81a88242002-10-26 15:22:42 +0000652 chip = simple_strtoul(argv[1], NULL, 16);
653
654 /*
655 * Address is always specified.
656 */
657 addr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700658 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +0200659 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000660 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000661
Simon Glassc11ddd42014-12-10 08:55:48 -0700662#ifdef CONFIG_DM_I2C
663 ret = i2c_get_cur_bus_chip(chip, &dev);
664 if (!ret && alen != -1)
665 ret = i2c_set_chip_offset_len(dev, alen);
666 if (ret)
667 return i2c_report_err(ret, I2C_ERR_WRITE);
668#endif
wdenk81a88242002-10-26 15:22:42 +0000669 /*
670 * Value to write is always specified.
671 */
672 byte = simple_strtoul(argv[3], NULL, 16);
673
674 /*
675 * Optional count
676 */
Timur Tabiff0215a2006-11-28 12:09:35 -0600677 if (argc == 5)
wdenk81a88242002-10-26 15:22:42 +0000678 count = simple_strtoul(argv[4], NULL, 16);
Timur Tabiff0215a2006-11-28 12:09:35 -0600679 else
wdenk81a88242002-10-26 15:22:42 +0000680 count = 1;
wdenk81a88242002-10-26 15:22:42 +0000681
682 while (count-- > 0) {
Simon Glassc11ddd42014-12-10 08:55:48 -0700683#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700684 ret = dm_i2c_write(dev, addr++, &byte, 1);
Simon Glassc11ddd42014-12-10 08:55:48 -0700685#else
686 ret = i2c_write(chip, addr++, alen, &byte, 1);
687#endif
688 if (ret)
Masahiro Yamada6b546f32015-02-05 13:50:26 +0900689 return i2c_report_err(ret, I2C_ERR_WRITE);
wdenk81a88242002-10-26 15:22:42 +0000690 /*
691 * Wait for the write to complete. The write can take
692 * up to 10mSec (we allow a little more time).
wdenk81a88242002-10-26 15:22:42 +0000693 */
m8a484c602005-08-12 21:16:13 +0200694/*
695 * No write delay with FRAM devices.
696 */
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200697#if !defined(CONFIG_SYS_I2C_FRAM)
wdenk81a88242002-10-26 15:22:42 +0000698 udelay(11000);
m8a484c602005-08-12 21:16:13 +0200699#endif
wdenk81a88242002-10-26 15:22:42 +0000700 }
701
Marek Vasut2476c8d2012-11-12 14:34:27 +0000702 return 0;
wdenk81a88242002-10-26 15:22:42 +0000703}
704
Marek Vasut2476c8d2012-11-12 14:34:27 +0000705/**
706 * do_i2c_crc() - Handle the "i2c crc32" command-line command
707 * @cmdtp: Command data struct pointer
708 * @flag: Command flag
709 * @argc: Command-line argument count
710 * @argv: Array of command-line arguments
711 *
712 * Calculate a CRC on memory
713 *
714 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
715 * on error.
wdenk81a88242002-10-26 15:22:42 +0000716 *
717 * Syntax:
Peter Tyser469cde42009-04-18 22:34:03 -0500718 * i2c crc32 {i2c_chip} {addr}{.0, .1, .2} {count}
wdenk81a88242002-10-26 15:22:42 +0000719 */
Simon Glassed38aef2020-05-10 11:40:03 -0600720static int do_i2c_crc(struct cmd_tbl *cmdtp, int flag, int argc,
721 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +0000722{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900723 uint chip;
wdenk81a88242002-10-26 15:22:42 +0000724 ulong addr;
Simon Glassc11ddd42014-12-10 08:55:48 -0700725 int alen;
wdenk81a88242002-10-26 15:22:42 +0000726 int count;
727 uchar byte;
728 ulong crc;
729 ulong err;
Simon Glassc11ddd42014-12-10 08:55:48 -0700730 int ret = 0;
731#ifdef CONFIG_DM_I2C
732 struct udevice *dev;
733#endif
wdenk81a88242002-10-26 15:22:42 +0000734
Wolfgang Denk3b683112010-07-17 01:06:04 +0200735 if (argc < 4)
Simon Glassa06dfc72011-12-10 08:44:01 +0000736 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000737
738 /*
Wolfgang Denka1be4762008-05-20 16:00:29 +0200739 * Chip is always specified.
740 */
wdenk81a88242002-10-26 15:22:42 +0000741 chip = simple_strtoul(argv[1], NULL, 16);
742
743 /*
744 * Address is always specified.
745 */
746 addr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700747 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +0200748 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000749 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000750
Simon Glassc11ddd42014-12-10 08:55:48 -0700751#ifdef CONFIG_DM_I2C
752 ret = i2c_get_cur_bus_chip(chip, &dev);
753 if (!ret && alen != -1)
754 ret = i2c_set_chip_offset_len(dev, alen);
755 if (ret)
756 return i2c_report_err(ret, I2C_ERR_READ);
757#endif
wdenk81a88242002-10-26 15:22:42 +0000758 /*
759 * Count is always specified
760 */
761 count = simple_strtoul(argv[3], NULL, 16);
762
763 printf ("CRC32 for %08lx ... %08lx ==> ", addr, addr + count - 1);
764 /*
765 * CRC a byte at a time. This is going to be slooow, but hey, the
766 * memories are small and slow too so hopefully nobody notices.
767 */
768 crc = 0;
769 err = 0;
Timur Tabiff0215a2006-11-28 12:09:35 -0600770 while (count-- > 0) {
Simon Glassc11ddd42014-12-10 08:55:48 -0700771#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700772 ret = dm_i2c_read(dev, addr, &byte, 1);
Simon Glassc11ddd42014-12-10 08:55:48 -0700773#else
774 ret = i2c_read(chip, addr, alen, &byte, 1);
775#endif
776 if (ret)
wdenk81a88242002-10-26 15:22:42 +0000777 err++;
Simon Glassc2226bf2019-11-14 12:57:15 -0700778 crc = crc32(crc, &byte, 1);
wdenk81a88242002-10-26 15:22:42 +0000779 addr++;
780 }
Timur Tabiff0215a2006-11-28 12:09:35 -0600781 if (err > 0)
Simon Glassc11ddd42014-12-10 08:55:48 -0700782 i2c_report_err(ret, I2C_ERR_READ);
Timur Tabiff0215a2006-11-28 12:09:35 -0600783 else
wdenk81a88242002-10-26 15:22:42 +0000784 printf ("%08lx\n", crc);
wdenk81a88242002-10-26 15:22:42 +0000785
786 return 0;
787}
788
Marek Vasut2476c8d2012-11-12 14:34:27 +0000789/**
790 * mod_i2c_mem() - Handle the "i2c mm" and "i2c nm" command-line command
791 * @cmdtp: Command data struct pointer
792 * @flag: Command flag
793 * @argc: Command-line argument count
794 * @argv: Array of command-line arguments
795 *
796 * Modify memory.
797 *
798 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
799 * on error.
wdenk81a88242002-10-26 15:22:42 +0000800 *
801 * Syntax:
Peter Tyser469cde42009-04-18 22:34:03 -0500802 * i2c mm{.b, .w, .l} {i2c_chip} {addr}{.0, .1, .2}
803 * i2c nm{.b, .w, .l} {i2c_chip} {addr}{.0, .1, .2}
wdenk81a88242002-10-26 15:22:42 +0000804 */
Simon Glassed38aef2020-05-10 11:40:03 -0600805static int mod_i2c_mem(struct cmd_tbl *cmdtp, int incrflag, int flag, int argc,
806 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +0000807{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +0900808 uint chip;
wdenk81a88242002-10-26 15:22:42 +0000809 ulong addr;
Simon Glassc11ddd42014-12-10 08:55:48 -0700810 int alen;
wdenk81a88242002-10-26 15:22:42 +0000811 ulong data;
812 int size = 1;
813 int nbytes;
Simon Glassc11ddd42014-12-10 08:55:48 -0700814 int ret;
815#ifdef CONFIG_DM_I2C
816 struct udevice *dev;
817#endif
wdenk81a88242002-10-26 15:22:42 +0000818
Wolfgang Denk3b683112010-07-17 01:06:04 +0200819 if (argc != 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000820 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000821
Simon Glass09007c42014-04-10 20:01:31 -0600822 bootretry_reset_cmd_timeout(); /* got a good command to get here */
wdenk81a88242002-10-26 15:22:42 +0000823 /*
824 * We use the last specified parameters, unless new ones are
825 * entered.
826 */
827 chip = i2c_mm_last_chip;
828 addr = i2c_mm_last_addr;
829 alen = i2c_mm_last_alen;
830
831 if ((flag & CMD_FLAG_REPEAT) == 0) {
832 /*
833 * New command specified. Check for a size specification.
834 * Defaults to byte if no or incorrect specification.
835 */
836 size = cmd_get_data_size(argv[0], 1);
837
838 /*
Wolfgang Denka1be4762008-05-20 16:00:29 +0200839 * Chip is always specified.
840 */
wdenk81a88242002-10-26 15:22:42 +0000841 chip = simple_strtoul(argv[1], NULL, 16);
842
843 /*
844 * Address is always specified.
845 */
846 addr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -0700847 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +0200848 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +0000849 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +0000850 }
851
Simon Glassc11ddd42014-12-10 08:55:48 -0700852#ifdef CONFIG_DM_I2C
853 ret = i2c_get_cur_bus_chip(chip, &dev);
854 if (!ret && alen != -1)
855 ret = i2c_set_chip_offset_len(dev, alen);
856 if (ret)
857 return i2c_report_err(ret, I2C_ERR_WRITE);
858#endif
859
wdenk81a88242002-10-26 15:22:42 +0000860 /*
861 * Print the address, followed by value. Then accept input for
862 * the next value. A non-converted value exits.
863 */
864 do {
865 printf("%08lx:", addr);
Simon Glassc11ddd42014-12-10 08:55:48 -0700866#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700867 ret = dm_i2c_read(dev, addr, (uchar *)&data, size);
Simon Glassc11ddd42014-12-10 08:55:48 -0700868#else
869 ret = i2c_read(chip, addr, alen, (uchar *)&data, size);
870#endif
871 if (ret)
Masahiro Yamada6b546f32015-02-05 13:50:26 +0900872 return i2c_report_err(ret, I2C_ERR_READ);
873
874 data = cpu_to_be32(data);
875 if (size == 1)
876 printf(" %02lx", (data >> 24) & 0x000000FF);
877 else if (size == 2)
878 printf(" %04lx", (data >> 16) & 0x0000FFFF);
879 else
880 printf(" %08lx", data);
wdenk81a88242002-10-26 15:22:42 +0000881
Simon Glassbe6aafc2014-04-10 20:01:27 -0600882 nbytes = cli_readline(" ? ");
wdenk81a88242002-10-26 15:22:42 +0000883 if (nbytes == 0) {
884 /*
885 * <CR> pressed as only input, don't modify current
886 * location and move to next.
887 */
888 if (incrflag)
889 addr += size;
890 nbytes = size;
Simon Glass09007c42014-04-10 20:01:31 -0600891 /* good enough to not time out */
892 bootretry_reset_cmd_timeout();
wdenk81a88242002-10-26 15:22:42 +0000893 }
894#ifdef CONFIG_BOOT_RETRY_TIME
Timur Tabiff0215a2006-11-28 12:09:35 -0600895 else if (nbytes == -2)
wdenk81a88242002-10-26 15:22:42 +0000896 break; /* timed out, exit the command */
wdenk81a88242002-10-26 15:22:42 +0000897#endif
898 else {
899 char *endp;
900
901 data = simple_strtoul(console_buffer, &endp, 16);
Timur Tabiff0215a2006-11-28 12:09:35 -0600902 if (size == 1)
wdenk81a88242002-10-26 15:22:42 +0000903 data = data << 24;
Timur Tabiff0215a2006-11-28 12:09:35 -0600904 else if (size == 2)
wdenk81a88242002-10-26 15:22:42 +0000905 data = data << 16;
wdenk81a88242002-10-26 15:22:42 +0000906 data = be32_to_cpu(data);
907 nbytes = endp - console_buffer;
908 if (nbytes) {
wdenk81a88242002-10-26 15:22:42 +0000909 /*
910 * good enough to not time out
911 */
Simon Glass09007c42014-04-10 20:01:31 -0600912 bootretry_reset_cmd_timeout();
Simon Glassc11ddd42014-12-10 08:55:48 -0700913#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700914 ret = dm_i2c_write(dev, addr, (uchar *)&data,
915 size);
Simon Glassc11ddd42014-12-10 08:55:48 -0700916#else
917 ret = i2c_write(chip, addr, alen,
918 (uchar *)&data, size);
919#endif
920 if (ret)
Masahiro Yamada6b546f32015-02-05 13:50:26 +0900921 return i2c_report_err(ret,
922 I2C_ERR_WRITE);
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200923#ifdef CONFIG_SYS_EEPROM_PAGE_WRITE_DELAY_MS
924 udelay(CONFIG_SYS_EEPROM_PAGE_WRITE_DELAY_MS * 1000);
wdenk2bb11052003-07-17 23:16:40 +0000925#endif
wdenk81a88242002-10-26 15:22:42 +0000926 if (incrflag)
927 addr += size;
928 }
929 }
930 } while (nbytes);
931
Peter Tyserf0d88812008-08-15 14:36:32 -0500932 i2c_mm_last_chip = chip;
933 i2c_mm_last_addr = addr;
934 i2c_mm_last_alen = alen;
wdenk81a88242002-10-26 15:22:42 +0000935
936 return 0;
937}
938
Marek Vasut2476c8d2012-11-12 14:34:27 +0000939/**
940 * do_i2c_probe() - Handle the "i2c probe" command-line command
941 * @cmdtp: Command data struct pointer
942 * @flag: Command flag
943 * @argc: Command-line argument count
944 * @argv: Array of command-line arguments
945 *
946 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
947 * on error.
948 *
wdenk81a88242002-10-26 15:22:42 +0000949 * Syntax:
Eric Nelson4de31682012-09-23 10:12:56 +0000950 * i2c probe {addr}
951 *
952 * Returns zero (success) if one or more I2C devices was found
wdenk81a88242002-10-26 15:22:42 +0000953 */
Simon Glassed38aef2020-05-10 11:40:03 -0600954static int do_i2c_probe(struct cmd_tbl *cmdtp, int flag, int argc,
955 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +0000956{
957 int j;
Eric Nelson4de31682012-09-23 10:12:56 +0000958 int addr = -1;
959 int found = 0;
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200960#if defined(CONFIG_SYS_I2C_NOPROBES)
wdenk81a88242002-10-26 15:22:42 +0000961 int k, skip;
Heiko Schochere0e55bc2012-01-16 21:12:24 +0000962 unsigned int bus = GET_BUS_NUM;
Ben Warren45657152006-09-07 16:50:54 -0400963#endif /* NOPROBES */
Simon Glassc11ddd42014-12-10 08:55:48 -0700964 int ret;
965#ifdef CONFIG_DM_I2C
966 struct udevice *bus, *dev;
967
968 if (i2c_get_cur_bus(&bus))
969 return CMD_RET_FAILURE;
970#endif
wdenk81a88242002-10-26 15:22:42 +0000971
Eric Nelson4de31682012-09-23 10:12:56 +0000972 if (argc == 2)
973 addr = simple_strtol(argv[1], 0, 16);
974
wdenk42c05472004-03-23 22:14:11 +0000975 puts ("Valid chip addresses:");
Timur Tabiff0215a2006-11-28 12:09:35 -0600976 for (j = 0; j < 128; j++) {
Eric Nelson4de31682012-09-23 10:12:56 +0000977 if ((0 <= addr) && (j != addr))
978 continue;
979
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200980#if defined(CONFIG_SYS_I2C_NOPROBES)
wdenk81a88242002-10-26 15:22:42 +0000981 skip = 0;
Axel Lin95f1d7e2013-06-22 21:56:35 +0800982 for (k = 0; k < ARRAY_SIZE(i2c_no_probes); k++) {
Timur Tabiff0215a2006-11-28 12:09:35 -0600983 if (COMPARE_BUS(bus, k) && COMPARE_ADDR(j, k)) {
wdenk81a88242002-10-26 15:22:42 +0000984 skip = 1;
985 break;
986 }
987 }
988 if (skip)
989 continue;
990#endif
Simon Glassc11ddd42014-12-10 08:55:48 -0700991#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -0700992 ret = dm_i2c_probe(bus, j, 0, &dev);
Simon Glassc11ddd42014-12-10 08:55:48 -0700993#else
994 ret = i2c_probe(j);
995#endif
996 if (ret == 0) {
wdenk81a88242002-10-26 15:22:42 +0000997 printf(" %02X", j);
Eric Nelson4de31682012-09-23 10:12:56 +0000998 found++;
999 }
wdenk81a88242002-10-26 15:22:42 +00001000 }
wdenk42c05472004-03-23 22:14:11 +00001001 putc ('\n');
wdenk81a88242002-10-26 15:22:42 +00001002
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +02001003#if defined(CONFIG_SYS_I2C_NOPROBES)
wdenk81a88242002-10-26 15:22:42 +00001004 puts ("Excluded chip addresses:");
Axel Lin95f1d7e2013-06-22 21:56:35 +08001005 for (k = 0; k < ARRAY_SIZE(i2c_no_probes); k++) {
Timur Tabiff0215a2006-11-28 12:09:35 -06001006 if (COMPARE_BUS(bus,k))
Ben Warren45657152006-09-07 16:50:54 -04001007 printf(" %02X", NO_PROBE_ADDR(k));
1008 }
wdenk42c05472004-03-23 22:14:11 +00001009 putc ('\n');
wdenk81a88242002-10-26 15:22:42 +00001010#endif
1011
Eric Nelson4de31682012-09-23 10:12:56 +00001012 return (0 == found);
wdenk81a88242002-10-26 15:22:42 +00001013}
1014
Marek Vasut2476c8d2012-11-12 14:34:27 +00001015/**
1016 * do_i2c_loop() - Handle the "i2c loop" command-line command
1017 * @cmdtp: Command data struct pointer
1018 * @flag: Command flag
1019 * @argc: Command-line argument count
1020 * @argv: Array of command-line arguments
1021 *
1022 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
1023 * on error.
1024 *
wdenk81a88242002-10-26 15:22:42 +00001025 * Syntax:
Peter Tyser469cde42009-04-18 22:34:03 -05001026 * i2c loop {i2c_chip} {addr}{.0, .1, .2} [{length}] [{delay}]
wdenk81a88242002-10-26 15:22:42 +00001027 * {length} - Number of bytes to read
1028 * {delay} - A DECIMAL number and defaults to 1000 uSec
1029 */
Simon Glassed38aef2020-05-10 11:40:03 -06001030static int do_i2c_loop(struct cmd_tbl *cmdtp, int flag, int argc,
1031 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +00001032{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +09001033 uint chip;
Simon Glassc11ddd42014-12-10 08:55:48 -07001034 int alen;
wdenk81a88242002-10-26 15:22:42 +00001035 uint addr;
1036 uint length;
1037 u_char bytes[16];
1038 int delay;
Simon Glassc11ddd42014-12-10 08:55:48 -07001039 int ret;
1040#ifdef CONFIG_DM_I2C
1041 struct udevice *dev;
1042#endif
wdenk81a88242002-10-26 15:22:42 +00001043
Wolfgang Denk3b683112010-07-17 01:06:04 +02001044 if (argc < 3)
Simon Glassa06dfc72011-12-10 08:44:01 +00001045 return CMD_RET_USAGE;
wdenk81a88242002-10-26 15:22:42 +00001046
1047 /*
1048 * Chip is always specified.
1049 */
1050 chip = simple_strtoul(argv[1], NULL, 16);
1051
1052 /*
1053 * Address is always specified.
1054 */
1055 addr = simple_strtoul(argv[2], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -07001056 alen = get_alen(argv[2], DEFAULT_ADDR_LEN);
Reinhard Meyerae62eb62010-08-25 14:41:16 +02001057 if (alen > 3)
Simon Glassa06dfc72011-12-10 08:44:01 +00001058 return CMD_RET_USAGE;
Simon Glassc11ddd42014-12-10 08:55:48 -07001059#ifdef CONFIG_DM_I2C
1060 ret = i2c_get_cur_bus_chip(chip, &dev);
1061 if (!ret && alen != -1)
1062 ret = i2c_set_chip_offset_len(dev, alen);
1063 if (ret)
1064 return i2c_report_err(ret, I2C_ERR_WRITE);
1065#endif
wdenk81a88242002-10-26 15:22:42 +00001066
1067 /*
1068 * Length is the number of objects, not number of bytes.
1069 */
1070 length = 1;
1071 length = simple_strtoul(argv[3], NULL, 16);
Timur Tabiff0215a2006-11-28 12:09:35 -06001072 if (length > sizeof(bytes))
wdenk81a88242002-10-26 15:22:42 +00001073 length = sizeof(bytes);
wdenk81a88242002-10-26 15:22:42 +00001074
1075 /*
1076 * The delay time (uSec) is optional.
1077 */
1078 delay = 1000;
Timur Tabiff0215a2006-11-28 12:09:35 -06001079 if (argc > 3)
wdenk81a88242002-10-26 15:22:42 +00001080 delay = simple_strtoul(argv[4], NULL, 10);
wdenk81a88242002-10-26 15:22:42 +00001081 /*
1082 * Run the loop...
1083 */
Timur Tabiff0215a2006-11-28 12:09:35 -06001084 while (1) {
Simon Glassc11ddd42014-12-10 08:55:48 -07001085#ifdef CONFIG_DM_I2C
Simon Glass7d722762015-01-12 18:02:07 -07001086 ret = dm_i2c_read(dev, addr, bytes, length);
Simon Glassc11ddd42014-12-10 08:55:48 -07001087#else
1088 ret = i2c_read(chip, addr, alen, bytes, length);
1089#endif
1090 if (ret)
1091 i2c_report_err(ret, I2C_ERR_READ);
wdenk81a88242002-10-26 15:22:42 +00001092 udelay(delay);
1093 }
1094
1095 /* NOTREACHED */
1096 return 0;
1097}
1098
wdenk81a88242002-10-26 15:22:42 +00001099/*
1100 * The SDRAM command is separately configured because many
1101 * (most?) embedded boards don't use SDRAM DIMMs.
Marek Vasut2476c8d2012-11-12 14:34:27 +00001102 *
1103 * FIXME: Document and probably move elsewhere!
wdenk81a88242002-10-26 15:22:42 +00001104 */
Jon Loeligerd76b5c12007-07-08 18:02:23 -05001105#if defined(CONFIG_CMD_SDRAM)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001106static void print_ddr2_tcyc (u_char const b)
1107{
1108 printf ("%d.", (b >> 4) & 0x0F);
1109 switch (b & 0x0F) {
1110 case 0x0:
1111 case 0x1:
1112 case 0x2:
1113 case 0x3:
1114 case 0x4:
1115 case 0x5:
1116 case 0x6:
1117 case 0x7:
1118 case 0x8:
1119 case 0x9:
1120 printf ("%d ns\n", b & 0x0F);
1121 break;
1122 case 0xA:
1123 puts ("25 ns\n");
1124 break;
1125 case 0xB:
1126 puts ("33 ns\n");
1127 break;
1128 case 0xC:
1129 puts ("66 ns\n");
1130 break;
1131 case 0xD:
1132 puts ("75 ns\n");
1133 break;
1134 default:
1135 puts ("?? ns\n");
1136 break;
1137 }
1138}
1139
1140static void decode_bits (u_char const b, char const *str[], int const do_once)
1141{
1142 u_char mask;
1143
1144 for (mask = 0x80; mask != 0x00; mask >>= 1, ++str) {
1145 if (b & mask) {
1146 puts (*str);
1147 if (do_once)
1148 return;
1149 }
1150 }
1151}
wdenk81a88242002-10-26 15:22:42 +00001152
1153/*
1154 * Syntax:
Peter Tyser469cde42009-04-18 22:34:03 -05001155 * i2c sdram {i2c_chip}
wdenk81a88242002-10-26 15:22:42 +00001156 */
Simon Glassed38aef2020-05-10 11:40:03 -06001157static int do_sdram(struct cmd_tbl *cmdtp, int flag, int argc,
1158 char *const argv[])
wdenk81a88242002-10-26 15:22:42 +00001159{
Michal Simek95e11f42016-02-15 11:58:37 +01001160 enum { unknown, EDO, SDRAM, DDR, DDR2, DDR3, DDR4 } type;
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001161
Masahiro Yamadab87abaf2014-12-20 03:34:23 +09001162 uint chip;
wdenk81a88242002-10-26 15:22:42 +00001163 u_char data[128];
1164 u_char cksum;
Nobuhiro Iwamatsu24e592c2017-12-01 14:39:40 +09001165 int j, ret;
1166#ifdef CONFIG_DM_I2C
1167 struct udevice *dev;
1168#endif
wdenk81a88242002-10-26 15:22:42 +00001169
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001170 static const char *decode_CAS_DDR2[] = {
1171 " TBD", " 6", " 5", " 4", " 3", " 2", " TBD", " TBD"
1172 };
1173
1174 static const char *decode_CAS_default[] = {
1175 " TBD", " 7", " 6", " 5", " 4", " 3", " 2", " 1"
1176 };
1177
1178 static const char *decode_CS_WE_default[] = {
1179 " TBD", " 6", " 5", " 4", " 3", " 2", " 1", " 0"
1180 };
1181
1182 static const char *decode_byte21_default[] = {
1183 " TBD (bit 7)\n",
1184 " Redundant row address\n",
1185 " Differential clock input\n",
1186 " Registerd DQMB inputs\n",
1187 " Buffered DQMB inputs\n",
1188 " On-card PLL\n",
1189 " Registered address/control lines\n",
1190 " Buffered address/control lines\n"
1191 };
1192
1193 static const char *decode_byte22_DDR2[] = {
1194 " TBD (bit 7)\n",
1195 " TBD (bit 6)\n",
1196 " TBD (bit 5)\n",
1197 " TBD (bit 4)\n",
1198 " TBD (bit 3)\n",
1199 " Supports partial array self refresh\n",
1200 " Supports 50 ohm ODT\n",
1201 " Supports weak driver\n"
1202 };
1203
1204 static const char *decode_row_density_DDR2[] = {
1205 "512 MiB", "256 MiB", "128 MiB", "16 GiB",
1206 "8 GiB", "4 GiB", "2 GiB", "1 GiB"
1207 };
1208
1209 static const char *decode_row_density_default[] = {
1210 "512 MiB", "256 MiB", "128 MiB", "64 MiB",
1211 "32 MiB", "16 MiB", "8 MiB", "4 MiB"
1212 };
1213
Wolfgang Denk3b683112010-07-17 01:06:04 +02001214 if (argc < 2)
Simon Glassa06dfc72011-12-10 08:44:01 +00001215 return CMD_RET_USAGE;
Wolfgang Denk3b683112010-07-17 01:06:04 +02001216
wdenk81a88242002-10-26 15:22:42 +00001217 /*
1218 * Chip is always specified.
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001219 */
1220 chip = simple_strtoul (argv[1], NULL, 16);
wdenk81a88242002-10-26 15:22:42 +00001221
Nobuhiro Iwamatsu24e592c2017-12-01 14:39:40 +09001222#ifdef CONFIG_DM_I2C
1223 ret = i2c_get_cur_bus_chip(chip, &dev);
1224 if (!ret)
1225 ret = dm_i2c_read(dev, 0, data, sizeof(data));
1226#else
1227 ret = i2c_read(chip, 0, 1, data, sizeof(data));
1228#endif
1229 if (ret) {
wdenk42c05472004-03-23 22:14:11 +00001230 puts ("No SDRAM Serial Presence Detect found.\n");
wdenk81a88242002-10-26 15:22:42 +00001231 return 1;
1232 }
1233
1234 cksum = 0;
1235 for (j = 0; j < 63; j++) {
1236 cksum += data[j];
1237 }
Timur Tabiff0215a2006-11-28 12:09:35 -06001238 if (cksum != data[63]) {
wdenk81a88242002-10-26 15:22:42 +00001239 printf ("WARNING: Configuration data checksum failure:\n"
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001240 " is 0x%02x, calculated 0x%02x\n", data[63], cksum);
wdenk81a88242002-10-26 15:22:42 +00001241 }
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001242 printf ("SPD data revision %d.%d\n",
wdenk81a88242002-10-26 15:22:42 +00001243 (data[62] >> 4) & 0x0F, data[62] & 0x0F);
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001244 printf ("Bytes used 0x%02X\n", data[0]);
1245 printf ("Serial memory size 0x%02X\n", 1 << data[1]);
1246
wdenk42c05472004-03-23 22:14:11 +00001247 puts ("Memory type ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001248 switch (data[2]) {
Larry Johnson330d19a2008-01-10 22:23:39 -05001249 case 2:
1250 type = EDO;
1251 puts ("EDO\n");
1252 break;
1253 case 4:
1254 type = SDRAM;
1255 puts ("SDRAM\n");
1256 break;
Michal Simek95e11f42016-02-15 11:58:37 +01001257 case 7:
1258 type = DDR;
1259 puts("DDR\n");
1260 break;
Larry Johnson330d19a2008-01-10 22:23:39 -05001261 case 8:
1262 type = DDR2;
1263 puts ("DDR2\n");
1264 break;
Michal Simek95e11f42016-02-15 11:58:37 +01001265 case 11:
1266 type = DDR3;
1267 puts("DDR3\n");
1268 break;
1269 case 12:
1270 type = DDR4;
1271 puts("DDR4\n");
1272 break;
Larry Johnson330d19a2008-01-10 22:23:39 -05001273 default:
1274 type = unknown;
1275 puts ("unknown\n");
1276 break;
wdenk81a88242002-10-26 15:22:42 +00001277 }
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001278
wdenk42c05472004-03-23 22:14:11 +00001279 puts ("Row address bits ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001280 if ((data[3] & 0x00F0) == 0)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001281 printf ("%d\n", data[3] & 0x0F);
Timur Tabiff0215a2006-11-28 12:09:35 -06001282 else
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001283 printf ("%d/%d\n", data[3] & 0x0F, (data[3] >> 4) & 0x0F);
1284
wdenk42c05472004-03-23 22:14:11 +00001285 puts ("Column address bits ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001286 if ((data[4] & 0x00F0) == 0)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001287 printf ("%d\n", data[4] & 0x0F);
Timur Tabiff0215a2006-11-28 12:09:35 -06001288 else
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001289 printf ("%d/%d\n", data[4] & 0x0F, (data[4] >> 4) & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001290
1291 switch (type) {
1292 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001293 printf ("Number of ranks %d\n",
1294 (data[5] & 0x07) + 1);
Larry Johnson330d19a2008-01-10 22:23:39 -05001295 break;
1296 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001297 printf ("Module rows %d\n", data[5]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001298 break;
1299 }
1300
1301 switch (type) {
1302 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001303 printf ("Module data width %d bits\n", data[6]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001304 break;
1305 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001306 printf ("Module data width %d bits\n",
1307 (data[7] << 8) | data[6]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001308 break;
1309 }
1310
wdenk42c05472004-03-23 22:14:11 +00001311 puts ("Interface signal levels ");
wdenk81a88242002-10-26 15:22:42 +00001312 switch(data[8]) {
Larry Johnson330d19a2008-01-10 22:23:39 -05001313 case 0: puts ("TTL 5.0 V\n"); break;
wdenk42c05472004-03-23 22:14:11 +00001314 case 1: puts ("LVTTL\n"); break;
Larry Johnson330d19a2008-01-10 22:23:39 -05001315 case 2: puts ("HSTL 1.5 V\n"); break;
1316 case 3: puts ("SSTL 3.3 V\n"); break;
1317 case 4: puts ("SSTL 2.5 V\n"); break;
1318 case 5: puts ("SSTL 1.8 V\n"); break;
wdenk42c05472004-03-23 22:14:11 +00001319 default: puts ("unknown\n"); break;
wdenk81a88242002-10-26 15:22:42 +00001320 }
Larry Johnson330d19a2008-01-10 22:23:39 -05001321
1322 switch (type) {
1323 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001324 printf ("SDRAM cycle time ");
1325 print_ddr2_tcyc (data[9]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001326 break;
1327 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001328 printf ("SDRAM cycle time %d.%d ns\n",
1329 (data[9] >> 4) & 0x0F, data[9] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001330 break;
1331 }
1332
1333 switch (type) {
1334 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001335 printf ("SDRAM access time 0.%d%d ns\n",
1336 (data[10] >> 4) & 0x0F, data[10] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001337 break;
1338 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001339 printf ("SDRAM access time %d.%d ns\n",
1340 (data[10] >> 4) & 0x0F, data[10] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001341 break;
1342 }
1343
wdenk42c05472004-03-23 22:14:11 +00001344 puts ("EDC configuration ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001345 switch (data[11]) {
wdenk42c05472004-03-23 22:14:11 +00001346 case 0: puts ("None\n"); break;
1347 case 1: puts ("Parity\n"); break;
1348 case 2: puts ("ECC\n"); break;
1349 default: puts ("unknown\n"); break;
wdenk81a88242002-10-26 15:22:42 +00001350 }
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001351
Timur Tabiff0215a2006-11-28 12:09:35 -06001352 if ((data[12] & 0x80) == 0)
wdenk42c05472004-03-23 22:14:11 +00001353 puts ("No self refresh, rate ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001354 else
wdenk42c05472004-03-23 22:14:11 +00001355 puts ("Self refresh, rate ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001356
wdenk81a88242002-10-26 15:22:42 +00001357 switch(data[12] & 0x7F) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001358 case 0: puts ("15.625 us\n"); break;
1359 case 1: puts ("3.9 us\n"); break;
1360 case 2: puts ("7.8 us\n"); break;
1361 case 3: puts ("31.3 us\n"); break;
1362 case 4: puts ("62.5 us\n"); break;
1363 case 5: puts ("125 us\n"); break;
wdenk42c05472004-03-23 22:14:11 +00001364 default: puts ("unknown\n"); break;
wdenk81a88242002-10-26 15:22:42 +00001365 }
Larry Johnson330d19a2008-01-10 22:23:39 -05001366
1367 switch (type) {
1368 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001369 printf ("SDRAM width (primary) %d\n", data[13]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001370 break;
1371 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001372 printf ("SDRAM width (primary) %d\n", data[13] & 0x7F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001373 if ((data[13] & 0x80) != 0) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001374 printf (" (second bank) %d\n",
1375 2 * (data[13] & 0x7F));
Larry Johnson330d19a2008-01-10 22:23:39 -05001376 }
1377 break;
1378 }
1379
1380 switch (type) {
1381 case DDR2:
1382 if (data[14] != 0)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001383 printf ("EDC width %d\n", data[14]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001384 break;
1385 default:
1386 if (data[14] != 0) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001387 printf ("EDC width %d\n",
1388 data[14] & 0x7F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001389
1390 if ((data[14] & 0x80) != 0) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001391 printf (" (second bank) %d\n",
1392 2 * (data[14] & 0x7F));
Larry Johnson330d19a2008-01-10 22:23:39 -05001393 }
1394 }
1395 break;
wdenk81a88242002-10-26 15:22:42 +00001396 }
Larry Johnson330d19a2008-01-10 22:23:39 -05001397
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001398 if (DDR2 != type) {
1399 printf ("Min clock delay, back-to-back random column addresses "
1400 "%d\n", data[15]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001401 }
1402
wdenk42c05472004-03-23 22:14:11 +00001403 puts ("Burst length(s) ");
1404 if (data[16] & 0x80) puts (" Page");
1405 if (data[16] & 0x08) puts (" 8");
1406 if (data[16] & 0x04) puts (" 4");
1407 if (data[16] & 0x02) puts (" 2");
1408 if (data[16] & 0x01) puts (" 1");
1409 putc ('\n');
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001410 printf ("Number of banks %d\n", data[17]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001411
1412 switch (type) {
1413 case DDR2:
1414 puts ("CAS latency(s) ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001415 decode_bits (data[18], decode_CAS_DDR2, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001416 putc ('\n');
1417 break;
1418 default:
1419 puts ("CAS latency(s) ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001420 decode_bits (data[18], decode_CAS_default, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001421 putc ('\n');
1422 break;
1423 }
1424
1425 if (DDR2 != type) {
1426 puts ("CS latency(s) ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001427 decode_bits (data[19], decode_CS_WE_default, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001428 putc ('\n');
1429 }
1430
1431 if (DDR2 != type) {
1432 puts ("WE latency(s) ");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001433 decode_bits (data[20], decode_CS_WE_default, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001434 putc ('\n');
1435 }
1436
1437 switch (type) {
1438 case DDR2:
1439 puts ("Module attributes:\n");
1440 if (data[21] & 0x80)
1441 puts (" TBD (bit 7)\n");
1442 if (data[21] & 0x40)
1443 puts (" Analysis probe installed\n");
1444 if (data[21] & 0x20)
1445 puts (" TBD (bit 5)\n");
1446 if (data[21] & 0x10)
1447 puts (" FET switch external enable\n");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001448 printf (" %d PLLs on DIMM\n", (data[21] >> 2) & 0x03);
Larry Johnson330d19a2008-01-10 22:23:39 -05001449 if (data[20] & 0x11) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001450 printf (" %d active registers on DIMM\n",
1451 (data[21] & 0x03) + 1);
Larry Johnson330d19a2008-01-10 22:23:39 -05001452 }
1453 break;
1454 default:
1455 puts ("Module attributes:\n");
1456 if (!data[21])
1457 puts (" (none)\n");
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001458 else
1459 decode_bits (data[21], decode_byte21_default, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001460 break;
1461 }
1462
1463 switch (type) {
1464 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001465 decode_bits (data[22], decode_byte22_DDR2, 0);
Larry Johnson330d19a2008-01-10 22:23:39 -05001466 break;
1467 default:
1468 puts ("Device attributes:\n");
1469 if (data[22] & 0x80) puts (" TBD (bit 7)\n");
1470 if (data[22] & 0x40) puts (" TBD (bit 6)\n");
1471 if (data[22] & 0x20) puts (" Upper Vcc tolerance 5%\n");
1472 else puts (" Upper Vcc tolerance 10%\n");
1473 if (data[22] & 0x10) puts (" Lower Vcc tolerance 5%\n");
1474 else puts (" Lower Vcc tolerance 10%\n");
1475 if (data[22] & 0x08) puts (" Supports write1/read burst\n");
1476 if (data[22] & 0x04) puts (" Supports precharge all\n");
1477 if (data[22] & 0x02) puts (" Supports auto precharge\n");
1478 if (data[22] & 0x01) puts (" Supports early RAS# precharge\n");
1479 break;
1480 }
1481
1482 switch (type) {
1483 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001484 printf ("SDRAM cycle time (2nd highest CAS latency) ");
1485 print_ddr2_tcyc (data[23]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001486 break;
1487 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001488 printf ("SDRAM cycle time (2nd highest CAS latency) %d."
1489 "%d ns\n", (data[23] >> 4) & 0x0F, data[23] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001490 break;
1491 }
1492
1493 switch (type) {
1494 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001495 printf ("SDRAM access from clock (2nd highest CAS latency) 0."
1496 "%d%d ns\n", (data[24] >> 4) & 0x0F, data[24] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001497 break;
1498 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001499 printf ("SDRAM access from clock (2nd highest CAS latency) %d."
1500 "%d ns\n", (data[24] >> 4) & 0x0F, data[24] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001501 break;
1502 }
1503
1504 switch (type) {
1505 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001506 printf ("SDRAM cycle time (3rd highest CAS latency) ");
1507 print_ddr2_tcyc (data[25]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001508 break;
1509 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001510 printf ("SDRAM cycle time (3rd highest CAS latency) %d."
1511 "%d ns\n", (data[25] >> 4) & 0x0F, data[25] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001512 break;
1513 }
1514
1515 switch (type) {
1516 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001517 printf ("SDRAM access from clock (3rd highest CAS latency) 0."
1518 "%d%d ns\n", (data[26] >> 4) & 0x0F, data[26] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001519 break;
1520 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001521 printf ("SDRAM access from clock (3rd highest CAS latency) %d."
1522 "%d ns\n", (data[26] >> 4) & 0x0F, data[26] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001523 break;
1524 }
1525
1526 switch (type) {
1527 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001528 printf ("Minimum row precharge %d.%02d ns\n",
1529 (data[27] >> 2) & 0x3F, 25 * (data[27] & 0x03));
Larry Johnson330d19a2008-01-10 22:23:39 -05001530 break;
1531 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001532 printf ("Minimum row precharge %d ns\n", data[27]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001533 break;
1534 }
1535
1536 switch (type) {
1537 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001538 printf ("Row active to row active min %d.%02d ns\n",
1539 (data[28] >> 2) & 0x3F, 25 * (data[28] & 0x03));
Larry Johnson330d19a2008-01-10 22:23:39 -05001540 break;
1541 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001542 printf ("Row active to row active min %d ns\n", data[28]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001543 break;
1544 }
1545
1546 switch (type) {
1547 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001548 printf ("RAS to CAS delay min %d.%02d ns\n",
1549 (data[29] >> 2) & 0x3F, 25 * (data[29] & 0x03));
Larry Johnson330d19a2008-01-10 22:23:39 -05001550 break;
1551 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001552 printf ("RAS to CAS delay min %d ns\n", data[29]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001553 break;
1554 }
1555
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001556 printf ("Minimum RAS pulse width %d ns\n", data[30]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001557
1558 switch (type) {
1559 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001560 puts ("Density of each row ");
1561 decode_bits (data[31], decode_row_density_DDR2, 1);
1562 putc ('\n');
Larry Johnson330d19a2008-01-10 22:23:39 -05001563 break;
1564 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001565 puts ("Density of each row ");
1566 decode_bits (data[31], decode_row_density_default, 1);
1567 putc ('\n');
Larry Johnson330d19a2008-01-10 22:23:39 -05001568 break;
1569 }
1570
1571 switch (type) {
1572 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001573 puts ("Command and Address setup ");
Larry Johnson330d19a2008-01-10 22:23:39 -05001574 if (data[32] >= 0xA0) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001575 printf ("1.%d%d ns\n",
1576 ((data[32] >> 4) & 0x0F) - 10, data[32] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001577 } else {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001578 printf ("0.%d%d ns\n",
1579 ((data[32] >> 4) & 0x0F), data[32] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001580 }
1581 break;
1582 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001583 printf ("Command and Address setup %c%d.%d ns\n",
1584 (data[32] & 0x80) ? '-' : '+',
1585 (data[32] >> 4) & 0x07, data[32] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001586 break;
1587 }
1588
1589 switch (type) {
1590 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001591 puts ("Command and Address hold ");
Larry Johnson330d19a2008-01-10 22:23:39 -05001592 if (data[33] >= 0xA0) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001593 printf ("1.%d%d ns\n",
1594 ((data[33] >> 4) & 0x0F) - 10, data[33] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001595 } else {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001596 printf ("0.%d%d ns\n",
1597 ((data[33] >> 4) & 0x0F), data[33] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001598 }
1599 break;
1600 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001601 printf ("Command and Address hold %c%d.%d ns\n",
1602 (data[33] & 0x80) ? '-' : '+',
1603 (data[33] >> 4) & 0x07, data[33] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001604 break;
1605 }
1606
1607 switch (type) {
1608 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001609 printf ("Data signal input setup 0.%d%d ns\n",
1610 (data[34] >> 4) & 0x0F, data[34] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001611 break;
1612 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001613 printf ("Data signal input setup %c%d.%d ns\n",
1614 (data[34] & 0x80) ? '-' : '+',
1615 (data[34] >> 4) & 0x07, data[34] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001616 break;
1617 }
1618
1619 switch (type) {
1620 case DDR2:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001621 printf ("Data signal input hold 0.%d%d ns\n",
1622 (data[35] >> 4) & 0x0F, data[35] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001623 break;
1624 default:
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001625 printf ("Data signal input hold %c%d.%d ns\n",
1626 (data[35] & 0x80) ? '-' : '+',
1627 (data[35] >> 4) & 0x07, data[35] & 0x0F);
Larry Johnson330d19a2008-01-10 22:23:39 -05001628 break;
1629 }
1630
wdenk42c05472004-03-23 22:14:11 +00001631 puts ("Manufacturer's JEDEC ID ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001632 for (j = 64; j <= 71; j++)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001633 printf ("%02X ", data[j]);
wdenk42c05472004-03-23 22:14:11 +00001634 putc ('\n');
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001635 printf ("Manufacturing Location %02X\n", data[72]);
wdenk42c05472004-03-23 22:14:11 +00001636 puts ("Manufacturer's Part Number ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001637 for (j = 73; j <= 90; j++)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001638 printf ("%02X ", data[j]);
wdenk42c05472004-03-23 22:14:11 +00001639 putc ('\n');
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001640 printf ("Revision Code %02X %02X\n", data[91], data[92]);
1641 printf ("Manufacturing Date %02X %02X\n", data[93], data[94]);
wdenk42c05472004-03-23 22:14:11 +00001642 puts ("Assembly Serial Number ");
Timur Tabiff0215a2006-11-28 12:09:35 -06001643 for (j = 95; j <= 98; j++)
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001644 printf ("%02X ", data[j]);
wdenk42c05472004-03-23 22:14:11 +00001645 putc ('\n');
wdenk81a88242002-10-26 15:22:42 +00001646
Larry Johnson330d19a2008-01-10 22:23:39 -05001647 if (DDR2 != type) {
Larry Johnson38c9b9b2008-01-11 23:26:18 -05001648 printf ("Speed rating PC%d\n",
1649 data[126] == 0x66 ? 66 : data[126]);
Larry Johnson330d19a2008-01-10 22:23:39 -05001650 }
wdenk81a88242002-10-26 15:22:42 +00001651 return 0;
1652}
Jon Loeligerd704d912007-07-10 11:02:44 -05001653#endif
wdenk81a88242002-10-26 15:22:42 +00001654
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001655/*
1656 * Syntax:
1657 * i2c edid {i2c_chip}
1658 */
1659#if defined(CONFIG_I2C_EDID)
Simon Glassed38aef2020-05-10 11:40:03 -06001660int do_edid(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[])
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001661{
Masahiro Yamadab87abaf2014-12-20 03:34:23 +09001662 uint chip;
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001663 struct edid1_info edid;
Simon Glassc11ddd42014-12-10 08:55:48 -07001664 int ret;
1665#ifdef CONFIG_DM_I2C
1666 struct udevice *dev;
1667#endif
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001668
1669 if (argc < 2) {
1670 cmd_usage(cmdtp);
1671 return 1;
1672 }
1673
1674 chip = simple_strtoul(argv[1], NULL, 16);
Simon Glassc11ddd42014-12-10 08:55:48 -07001675#ifdef CONFIG_DM_I2C
1676 ret = i2c_get_cur_bus_chip(chip, &dev);
1677 if (!ret)
Simon Glass7d722762015-01-12 18:02:07 -07001678 ret = dm_i2c_read(dev, 0, (uchar *)&edid, sizeof(edid));
Simon Glassc11ddd42014-12-10 08:55:48 -07001679#else
1680 ret = i2c_read(chip, 0, 1, (uchar *)&edid, sizeof(edid));
1681#endif
1682 if (ret)
1683 return i2c_report_err(ret, I2C_ERR_READ);
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001684
1685 if (edid_check_info(&edid)) {
1686 puts("Content isn't valid EDID.\n");
1687 return 1;
1688 }
1689
1690 edid_print_info(&edid);
1691 return 0;
1692
1693}
1694#endif /* CONFIG_I2C_EDID */
1695
Simon Glassb8c2f882015-05-04 11:30:57 -06001696#ifdef CONFIG_DM_I2C
1697static void show_bus(struct udevice *bus)
1698{
1699 struct udevice *dev;
1700
1701 printf("Bus %d:\t%s", bus->req_seq, bus->name);
1702 if (device_active(bus))
1703 printf(" (active %d)", bus->seq);
1704 printf("\n");
1705 for (device_find_first_child(bus, &dev);
1706 dev;
1707 device_find_next_child(&dev)) {
1708 struct dm_i2c_chip *chip = dev_get_parent_platdata(dev);
1709
1710 printf(" %02x: %s, offset len %x, flags %x\n",
1711 chip->chip_addr, dev->name, chip->offset_len,
1712 chip->flags);
1713 }
1714}
1715#endif
1716
Marek Vasut2476c8d2012-11-12 14:34:27 +00001717/**
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001718 * do_i2c_show_bus() - Handle the "i2c bus" command-line command
Marek Vasut2476c8d2012-11-12 14:34:27 +00001719 * @cmdtp: Command data struct pointer
1720 * @flag: Command flag
1721 * @argc: Command-line argument count
1722 * @argv: Array of command-line arguments
1723 *
1724 * Returns zero always.
1725 */
Simon Glassb8c2f882015-05-04 11:30:57 -06001726#if defined(CONFIG_SYS_I2C) || defined(CONFIG_DM_I2C)
Simon Glassed38aef2020-05-10 11:40:03 -06001727static int do_i2c_show_bus(struct cmd_tbl *cmdtp, int flag, int argc,
1728 char *const argv[])
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001729{
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001730 if (argc == 1) {
1731 /* show all busses */
Simon Glassb8c2f882015-05-04 11:30:57 -06001732#ifdef CONFIG_DM_I2C
1733 struct udevice *bus;
1734 struct uclass *uc;
1735 int ret;
1736
1737 ret = uclass_get(UCLASS_I2C, &uc);
1738 if (ret)
1739 return CMD_RET_FAILURE;
1740 uclass_foreach_dev(bus, uc)
1741 show_bus(bus);
1742#else
1743 int i;
1744
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001745 for (i = 0; i < CONFIG_SYS_NUM_I2C_BUSES; i++) {
1746 printf("Bus %d:\t%s", i, I2C_ADAP_NR(i)->name);
1747#ifndef CONFIG_SYS_I2C_DIRECT_BUS
Simon Glassb8c2f882015-05-04 11:30:57 -06001748 int j;
1749
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001750 for (j = 0; j < CONFIG_SYS_I2C_MAX_HOPS; j++) {
1751 if (i2c_bus[i].next_hop[j].chip == 0)
1752 break;
1753 printf("->%s@0x%2x:%d",
1754 i2c_bus[i].next_hop[j].mux.name,
1755 i2c_bus[i].next_hop[j].chip,
1756 i2c_bus[i].next_hop[j].channel);
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001757 }
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001758#endif
1759 printf("\n");
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001760 }
Simon Glassb8c2f882015-05-04 11:30:57 -06001761#endif
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001762 } else {
Simon Glassb8c2f882015-05-04 11:30:57 -06001763 int i;
1764
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001765 /* show specific bus */
1766 i = simple_strtoul(argv[1], NULL, 10);
Simon Glassb8c2f882015-05-04 11:30:57 -06001767#ifdef CONFIG_DM_I2C
1768 struct udevice *bus;
1769 int ret;
1770
1771 ret = uclass_get_device_by_seq(UCLASS_I2C, i, &bus);
1772 if (ret) {
1773 printf("Invalid bus %d: err=%d\n", i, ret);
1774 return CMD_RET_FAILURE;
1775 }
1776 show_bus(bus);
1777#else
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001778 if (i >= CONFIG_SYS_NUM_I2C_BUSES) {
1779 printf("Invalid bus %d\n", i);
1780 return -1;
1781 }
1782 printf("Bus %d:\t%s", i, I2C_ADAP_NR(i)->name);
1783#ifndef CONFIG_SYS_I2C_DIRECT_BUS
Simon Glassb8c2f882015-05-04 11:30:57 -06001784 int j;
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001785 for (j = 0; j < CONFIG_SYS_I2C_MAX_HOPS; j++) {
1786 if (i2c_bus[i].next_hop[j].chip == 0)
1787 break;
1788 printf("->%s@0x%2x:%d",
1789 i2c_bus[i].next_hop[j].mux.name,
1790 i2c_bus[i].next_hop[j].chip,
1791 i2c_bus[i].next_hop[j].channel);
1792 }
1793#endif
1794 printf("\n");
Simon Glassb8c2f882015-05-04 11:30:57 -06001795#endif
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001796 }
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001797
1798 return 0;
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001799}
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001800#endif
Heiko Schocher6ee861b2008-10-15 09:39:47 +02001801
Marek Vasut2476c8d2012-11-12 14:34:27 +00001802/**
1803 * do_i2c_bus_num() - Handle the "i2c dev" command-line command
1804 * @cmdtp: Command data struct pointer
1805 * @flag: Command flag
1806 * @argc: Command-line argument count
1807 * @argv: Array of command-line arguments
1808 *
1809 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
1810 * on error.
1811 */
Simon Glassc11ddd42014-12-10 08:55:48 -07001812#if defined(CONFIG_SYS_I2C) || defined(CONFIG_I2C_MULTI_BUS) || \
1813 defined(CONFIG_DM_I2C)
Simon Glassed38aef2020-05-10 11:40:03 -06001814static int do_i2c_bus_num(struct cmd_tbl *cmdtp, int flag, int argc,
1815 char *const argv[])
Ben Warren45657152006-09-07 16:50:54 -04001816{
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001817 int ret = 0;
Simon Glassc11ddd42014-12-10 08:55:48 -07001818 int bus_no;
Ben Warren45657152006-09-07 16:50:54 -04001819
Simon Glassc11ddd42014-12-10 08:55:48 -07001820 if (argc == 1) {
Timur Tabiff0215a2006-11-28 12:09:35 -06001821 /* querying current setting */
Simon Glassc11ddd42014-12-10 08:55:48 -07001822#ifdef CONFIG_DM_I2C
1823 struct udevice *bus;
1824
1825 if (!i2c_get_cur_bus(&bus))
1826 bus_no = bus->seq;
1827 else
1828 bus_no = -1;
1829#else
1830 bus_no = i2c_get_bus_num();
1831#endif
1832 printf("Current bus is %d\n", bus_no);
1833 } else {
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001834 bus_no = simple_strtoul(argv[1], NULL, 10);
Heiko Schocher52fe2bf2013-08-23 09:39:16 +02001835#if defined(CONFIG_SYS_I2C)
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001836 if (bus_no >= CONFIG_SYS_NUM_I2C_BUSES) {
1837 printf("Invalid bus %d\n", bus_no);
1838 return -1;
1839 }
Heiko Schocher52fe2bf2013-08-23 09:39:16 +02001840#endif
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001841 printf("Setting bus to %d\n", bus_no);
Simon Glass7d722762015-01-12 18:02:07 -07001842#ifdef CONFIG_DM_I2C
1843 ret = cmd_i2c_set_bus_num(bus_no);
1844#else
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001845 ret = i2c_set_bus_num(bus_no);
Simon Glass7d722762015-01-12 18:02:07 -07001846#endif
Timur Tabiff0215a2006-11-28 12:09:35 -06001847 if (ret)
Ben Warren45657152006-09-07 16:50:54 -04001848 printf("Failure changing bus number (%d)\n", ret);
Ben Warren45657152006-09-07 16:50:54 -04001849 }
Simon Glass63f57852015-12-29 05:22:50 -07001850
1851 return ret ? CMD_RET_FAILURE : 0;
Ben Warren45657152006-09-07 16:50:54 -04001852}
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001853#endif /* defined(CONFIG_SYS_I2C) */
Ben Warren45657152006-09-07 16:50:54 -04001854
Marek Vasut2476c8d2012-11-12 14:34:27 +00001855/**
1856 * do_i2c_bus_speed() - Handle the "i2c speed" command-line command
1857 * @cmdtp: Command data struct pointer
1858 * @flag: Command flag
1859 * @argc: Command-line argument count
1860 * @argv: Array of command-line arguments
1861 *
1862 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
1863 * on error.
1864 */
Simon Glassed38aef2020-05-10 11:40:03 -06001865static int do_i2c_bus_speed(struct cmd_tbl *cmdtp, int flag, int argc,
1866 char *const argv[])
Ben Warren45657152006-09-07 16:50:54 -04001867{
1868 int speed, ret=0;
1869
Simon Glassc11ddd42014-12-10 08:55:48 -07001870#ifdef CONFIG_DM_I2C
1871 struct udevice *bus;
1872
1873 if (i2c_get_cur_bus(&bus))
1874 return 1;
1875#endif
1876 if (argc == 1) {
1877#ifdef CONFIG_DM_I2C
Simon Glasse4e8ff22015-02-05 21:41:32 -07001878 speed = dm_i2c_get_bus_speed(bus);
Simon Glassc11ddd42014-12-10 08:55:48 -07001879#else
1880 speed = i2c_get_bus_speed();
1881#endif
Timur Tabiff0215a2006-11-28 12:09:35 -06001882 /* querying current speed */
Simon Glassc11ddd42014-12-10 08:55:48 -07001883 printf("Current bus speed=%d\n", speed);
1884 } else {
Ben Warren45657152006-09-07 16:50:54 -04001885 speed = simple_strtoul(argv[1], NULL, 10);
1886 printf("Setting bus speed to %d Hz\n", speed);
Simon Glassc11ddd42014-12-10 08:55:48 -07001887#ifdef CONFIG_DM_I2C
Simon Glasse4e8ff22015-02-05 21:41:32 -07001888 ret = dm_i2c_set_bus_speed(bus, speed);
Simon Glassc11ddd42014-12-10 08:55:48 -07001889#else
Ben Warren45657152006-09-07 16:50:54 -04001890 ret = i2c_set_bus_speed(speed);
Simon Glassc11ddd42014-12-10 08:55:48 -07001891#endif
Timur Tabiff0215a2006-11-28 12:09:35 -06001892 if (ret)
Ben Warren45657152006-09-07 16:50:54 -04001893 printf("Failure changing bus speed (%d)\n", ret);
Ben Warren45657152006-09-07 16:50:54 -04001894 }
Simon Glass63f57852015-12-29 05:22:50 -07001895
1896 return ret ? CMD_RET_FAILURE : 0;
Ben Warren45657152006-09-07 16:50:54 -04001897}
1898
Marek Vasut2476c8d2012-11-12 14:34:27 +00001899/**
1900 * do_i2c_mm() - Handle the "i2c mm" command-line command
1901 * @cmdtp: Command data struct pointer
1902 * @flag: Command flag
1903 * @argc: Command-line argument count
1904 * @argv: Array of command-line arguments
1905 *
1906 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
1907 * on error.
1908 */
Simon Glassed38aef2020-05-10 11:40:03 -06001909static int do_i2c_mm(struct cmd_tbl *cmdtp, int flag, int argc,
1910 char *const argv[])
Ben Warren45657152006-09-07 16:50:54 -04001911{
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001912 return mod_i2c_mem (cmdtp, 1, flag, argc, argv);
1913}
1914
Marek Vasut2476c8d2012-11-12 14:34:27 +00001915/**
1916 * do_i2c_nm() - Handle the "i2c nm" command-line command
1917 * @cmdtp: Command data struct pointer
1918 * @flag: Command flag
1919 * @argc: Command-line argument count
1920 * @argv: Array of command-line arguments
1921 *
1922 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
1923 * on error.
1924 */
Simon Glassed38aef2020-05-10 11:40:03 -06001925static int do_i2c_nm(struct cmd_tbl *cmdtp, int flag, int argc,
1926 char *const argv[])
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001927{
1928 return mod_i2c_mem (cmdtp, 0, flag, argc, argv);
1929}
Peter Tyser4ff03cf2009-04-18 22:34:04 -05001930
Marek Vasut2476c8d2012-11-12 14:34:27 +00001931/**
1932 * do_i2c_reset() - Handle the "i2c reset" command-line command
1933 * @cmdtp: Command data struct pointer
1934 * @flag: Command flag
1935 * @argc: Command-line argument count
1936 * @argv: Array of command-line arguments
1937 *
1938 * Returns zero always.
1939 */
Simon Glassed38aef2020-05-10 11:40:03 -06001940static int do_i2c_reset(struct cmd_tbl *cmdtp, int flag, int argc,
1941 char *const argv[])
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001942{
Simon Glassc11ddd42014-12-10 08:55:48 -07001943#if defined(CONFIG_DM_I2C)
1944 struct udevice *bus;
1945
1946 if (i2c_get_cur_bus(&bus))
1947 return CMD_RET_FAILURE;
1948 if (i2c_deblock(bus)) {
1949 printf("Error: Not supported by the driver\n");
1950 return CMD_RET_FAILURE;
1951 }
1952#elif defined(CONFIG_SYS_I2C)
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001953 i2c_init(I2C_ADAP->speed, I2C_ADAP->slaveaddr);
1954#else
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001955 i2c_init(CONFIG_SYS_I2C_SPEED, CONFIG_SYS_I2C_SLAVE);
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001956#endif
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001957 return 0;
1958}
1959
Simon Glassed38aef2020-05-10 11:40:03 -06001960static struct cmd_tbl cmd_i2c_sub[] = {
Simon Glassb8c2f882015-05-04 11:30:57 -06001961#if defined(CONFIG_SYS_I2C) || defined(CONFIG_DM_I2C)
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001962 U_BOOT_CMD_MKENT(bus, 1, 1, do_i2c_show_bus, "", ""),
Heiko Schochera4ea4452012-10-25 10:32:14 +02001963#endif
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001964 U_BOOT_CMD_MKENT(crc32, 3, 1, do_i2c_crc, "", ""),
Heiko Schochere0e55bc2012-01-16 21:12:24 +00001965#if defined(CONFIG_SYS_I2C) || \
Simon Glassc11ddd42014-12-10 08:55:48 -07001966 defined(CONFIG_I2C_MULTI_BUS) || defined(CONFIG_DM_I2C)
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001967 U_BOOT_CMD_MKENT(dev, 1, 1, do_i2c_bus_num, "", ""),
Ben Warren45657152006-09-07 16:50:54 -04001968#endif /* CONFIG_I2C_MULTI_BUS */
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00001969#if defined(CONFIG_I2C_EDID)
1970 U_BOOT_CMD_MKENT(edid, 1, 1, do_edid, "", ""),
1971#endif /* CONFIG_I2C_EDID */
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001972 U_BOOT_CMD_MKENT(loop, 3, 1, do_i2c_loop, "", ""),
1973 U_BOOT_CMD_MKENT(md, 3, 1, do_i2c_md, "", ""),
1974 U_BOOT_CMD_MKENT(mm, 2, 1, do_i2c_mm, "", ""),
1975 U_BOOT_CMD_MKENT(mw, 3, 1, do_i2c_mw, "", ""),
1976 U_BOOT_CMD_MKENT(nm, 2, 1, do_i2c_nm, "", ""),
1977 U_BOOT_CMD_MKENT(probe, 0, 1, do_i2c_probe, "", ""),
Frans Meulenbroeks290eefb2010-02-25 10:12:16 +01001978 U_BOOT_CMD_MKENT(read, 5, 1, do_i2c_read, "", ""),
Lubomir Popov56dae702015-01-30 19:56:04 +02001979 U_BOOT_CMD_MKENT(write, 6, 0, do_i2c_write, "", ""),
Simon Glassc11ddd42014-12-10 08:55:48 -07001980#ifdef CONFIG_DM_I2C
1981 U_BOOT_CMD_MKENT(flags, 2, 1, do_i2c_flags, "", ""),
Simon Glassa44d5df2015-08-22 18:31:33 -06001982 U_BOOT_CMD_MKENT(olen, 2, 1, do_i2c_olen, "", ""),
Simon Glassc11ddd42014-12-10 08:55:48 -07001983#endif
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001984 U_BOOT_CMD_MKENT(reset, 0, 1, do_i2c_reset, "", ""),
Jon Loeligerd76b5c12007-07-08 18:02:23 -05001985#if defined(CONFIG_CMD_SDRAM)
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001986 U_BOOT_CMD_MKENT(sdram, 1, 1, do_sdram, "", ""),
Jon Loeligerd704d912007-07-10 11:02:44 -05001987#endif
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01001988 U_BOOT_CMD_MKENT(speed, 1, 1, do_i2c_bus_speed, "", ""),
1989};
1990
Michal Simek28990ed2015-12-04 16:56:57 +01001991static __maybe_unused void i2c_reloc(void)
1992{
1993 static int relocated;
1994
1995 if (!relocated) {
1996 fixup_cmdtable(cmd_i2c_sub, ARRAY_SIZE(cmd_i2c_sub));
1997 relocated = 1;
1998 };
Heiko Schocheraeb29912010-09-17 13:10:39 +02001999}
Heiko Schocheraeb29912010-09-17 13:10:39 +02002000
Marek Vasut2476c8d2012-11-12 14:34:27 +00002001/**
2002 * do_i2c() - Handle the "i2c" command-line command
2003 * @cmdtp: Command data struct pointer
2004 * @flag: Command flag
2005 * @argc: Command-line argument count
2006 * @argv: Array of command-line arguments
2007 *
2008 * Returns zero on success, CMD_RET_USAGE in case of misuse and negative
2009 * on error.
2010 */
Simon Glassed38aef2020-05-10 11:40:03 -06002011static int do_i2c(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[])
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01002012{
Simon Glassed38aef2020-05-10 11:40:03 -06002013 struct cmd_tbl *c;
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01002014
Michal Simek28990ed2015-12-04 16:56:57 +01002015#ifdef CONFIG_NEEDS_MANUAL_RELOC
2016 i2c_reloc();
2017#endif
2018
Heiko Schocher167ad642010-09-17 13:10:35 +02002019 if (argc < 2)
Simon Glassa06dfc72011-12-10 08:44:01 +00002020 return CMD_RET_USAGE;
Heiko Schocher167ad642010-09-17 13:10:35 +02002021
Frans Meulenbroeksa5b95222010-02-25 10:12:14 +01002022 /* Strip off leading 'i2c' command argument */
2023 argc--;
2024 argv++;
2025
2026 c = find_cmd_tbl(argv[0], &cmd_i2c_sub[0], ARRAY_SIZE(cmd_i2c_sub));
2027
Wolfgang Denk3b683112010-07-17 01:06:04 +02002028 if (c)
Simon Glassa06dfc72011-12-10 08:44:01 +00002029 return c->cmd(cmdtp, flag, argc, argv);
Wolfgang Denk3b683112010-07-17 01:06:04 +02002030 else
Simon Glassa06dfc72011-12-10 08:44:01 +00002031 return CMD_RET_USAGE;
Ben Warren45657152006-09-07 16:50:54 -04002032}
wdenk57b2d802003-06-27 21:31:46 +00002033
2034/***************************************************/
Kim Phillipsdc00a682012-10-29 13:34:31 +00002035#ifdef CONFIG_SYS_LONGHELP
2036static char i2c_help_text[] =
Simon Glassb8c2f882015-05-04 11:30:57 -06002037#if defined(CONFIG_SYS_I2C) || defined(CONFIG_DM_I2C)
Heiko Schochere0e55bc2012-01-16 21:12:24 +00002038 "bus [muxtype:muxaddr:muxchannel] - show I2C bus info\n"
Christoph Muellnerfc574be2018-12-04 11:27:18 +01002039 "i2c " /* That's the prefix for the crc32 command below. */
Heiko Schochera4ea4452012-10-25 10:32:14 +02002040#endif
Frans Meulenbroeks1c704022010-02-25 10:12:15 +01002041 "crc32 chip address[.0, .1, .2] count - compute CRC32 checksum\n"
Heiko Schochere0e55bc2012-01-16 21:12:24 +00002042#if defined(CONFIG_SYS_I2C) || \
Simon Glassc11ddd42014-12-10 08:55:48 -07002043 defined(CONFIG_I2C_MULTI_BUS) || defined(CONFIG_DM_I2C)
Peter Tyser8d4b8b52008-10-01 12:25:04 -05002044 "i2c dev [dev] - show or set current I2C bus\n"
Matthias Fuchsdd6cffc2007-03-08 16:25:47 +01002045#endif /* CONFIG_I2C_MULTI_BUS */
Tom Wai-Hong Tam6664f202012-12-05 14:46:40 +00002046#if defined(CONFIG_I2C_EDID)
2047 "i2c edid chip - print EDID configuration information\n"
2048#endif /* CONFIG_I2C_EDID */
Frans Meulenbroeks1c704022010-02-25 10:12:15 +01002049 "i2c loop chip address[.0, .1, .2] [# of objects] - looping read of device\n"
Matthias Fuchsdd6cffc2007-03-08 16:25:47 +01002050 "i2c md chip address[.0, .1, .2] [# of objects] - read from I2C device\n"
2051 "i2c mm chip address[.0, .1, .2] - write to I2C device (auto-incrementing)\n"
2052 "i2c mw chip address[.0, .1, .2] value [count] - write to I2C device (fill)\n"
2053 "i2c nm chip address[.0, .1, .2] - write to I2C device (constant address)\n"
Eric Nelson4de31682012-09-23 10:12:56 +00002054 "i2c probe [address] - test for and show device(s) on the I2C bus\n"
Simon Glassc11ddd42014-12-10 08:55:48 -07002055 "i2c read chip address[.0, .1, .2] length memaddress - read to memory\n"
Lubomir Popov56dae702015-01-30 19:56:04 +02002056 "i2c write memaddress chip address[.0, .1, .2] length [-s] - write memory\n"
2057 " to I2C; the -s option selects bulk write in a single transaction\n"
Simon Glassc11ddd42014-12-10 08:55:48 -07002058#ifdef CONFIG_DM_I2C
2059 "i2c flags chip [flags] - set or get chip flags\n"
Simon Glassa44d5df2015-08-22 18:31:33 -06002060 "i2c olen chip [offset_length] - set or get chip offset length\n"
Simon Glassc11ddd42014-12-10 08:55:48 -07002061#endif
Heiko Schocher017568e2008-10-15 09:33:30 +02002062 "i2c reset - re-init the I2C Controller\n"
Jon Loeligerd76b5c12007-07-08 18:02:23 -05002063#if defined(CONFIG_CMD_SDRAM)
Frans Meulenbroeks1c704022010-02-25 10:12:15 +01002064 "i2c sdram chip - print SDRAM configuration information\n"
Jon Loeligerd704d912007-07-10 11:02:44 -05002065#endif
Kim Phillipsdc00a682012-10-29 13:34:31 +00002066 "i2c speed [speed] - show or set I2C bus speed";
2067#endif
2068
2069U_BOOT_CMD(
Lubomir Popov56dae702015-01-30 19:56:04 +02002070 i2c, 7, 1, do_i2c,
Kim Phillipsdc00a682012-10-29 13:34:31 +00002071 "I2C sub-system",
2072 i2c_help_text
Matthias Fuchsdd6cffc2007-03-08 16:25:47 +01002073);