1 /*
2  * Copyright (c) 2022 Google Inc
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <stdlib.h>
8 #include <string.h>
9 
10 #include <zephyr/devicetree.h>
11 #include <zephyr/drivers/bbram.h>
12 #include <zephyr/kernel.h>
13 #include <zephyr/shell/shell.h>
14 #include <zephyr/sys/util.h>
15 
16 /* Buffer is only needed for bytes that follow command, device and address. */
17 #define BUF_ARRAY_CNT (CONFIG_SHELL_ARGC_MAX - 3)
18 
parse_ul(const char * str,unsigned long * result)19 static inline int parse_ul(const char *str, unsigned long *result)
20 {
21 	char *end;
22 	unsigned long val;
23 
24 	val = strtoul(str, &end, 0);
25 
26 	if (*str == '\0' || *end != '\0') {
27 		return -EINVAL;
28 	}
29 
30 	*result = val;
31 	return 0;
32 }
33 
parse_u32(const char * str,uint32_t * result)34 static inline int parse_u32(const char *str, uint32_t *result)
35 {
36 	unsigned long val;
37 
38 	if (parse_ul(str, &val) || val > 0xffffffff) {
39 		return -EINVAL;
40 	}
41 	*result = (uint32_t)val;
42 	return 0;
43 }
44 
parse_u8(const char * str,uint8_t * result)45 static inline int parse_u8(const char *str, uint8_t *result)
46 {
47 	unsigned long val;
48 
49 	if (parse_ul(str, &val) || val > 0xff) {
50 		return -EINVAL;
51 	}
52 	*result = (uint8_t)val;
53 	return 0;
54 }
55 
parse_device(const struct shell * sh,size_t argc,char * argv[],const struct device ** bbram_dev)56 static inline int parse_device(const struct shell *sh, size_t argc, char *argv[],
57 			       const struct device **bbram_dev)
58 {
59 	if (argc < 2) {
60 		shell_error(sh, "Missing BBRAM device");
61 		return -EINVAL;
62 	}
63 
64 	*bbram_dev = shell_device_get_binding(argv[1]);
65 	if (!*bbram_dev) {
66 		shell_error(sh, "Given BBRAM device was not found");
67 		return -ENODEV;
68 	}
69 
70 	return 0;
71 }
72 
cmd_read(const struct shell * sh,size_t argc,char * argv[])73 static int cmd_read(const struct shell *sh, size_t argc, char *argv[])
74 {
75 	const struct device *bbram_dev;
76 	uint32_t addr;
77 	size_t size;
78 	int part_size, ret;
79 
80 	ret = parse_device(sh, argc, argv, &bbram_dev);
81 	if (ret) {
82 		return ret;
83 	}
84 
85 	if (argc < 3) {
86 		/* Dump whole BBRAM if address not provided. */
87 		addr = 0;
88 		ret = bbram_get_size(bbram_dev, &size);
89 		if (ret < 0) {
90 			shell_error(sh, "Can't get BBRAM size: %d", ret);
91 			return -EIO;
92 		}
93 	} else {
94 		/* Parse address if provided. */
95 		ret = parse_u32(argv[2], &addr);
96 		if (ret) {
97 			return ret;
98 		}
99 
100 		/* If size not provided read one byte. */
101 		size = 1;
102 
103 		if (argc >= 4) {
104 			/* Parse size if provided. */
105 			ret = parse_u32(argv[3], &size);
106 			if (ret) {
107 				return ret;
108 			}
109 		}
110 	}
111 
112 	for (int cnt = 0; cnt < size; cnt += part_size) {
113 		uint8_t data[SHELL_HEXDUMP_BYTES_IN_LINE];
114 
115 		part_size = MIN(size - cnt, SHELL_HEXDUMP_BYTES_IN_LINE);
116 		ret = bbram_read(bbram_dev, addr, part_size, data);
117 		if (ret != 0) {
118 			shell_error(sh, "BBRAM read error: %d", ret);
119 			return -EIO;
120 		}
121 		shell_hexdump_line(sh, addr, data, part_size);
122 		addr += part_size;
123 	}
124 
125 	shell_print(sh, "");
126 	return 0;
127 }
128 
cmd_write(const struct shell * sh,size_t argc,char * argv[])129 static int cmd_write(const struct shell *sh, size_t argc, char *argv[])
130 {
131 	const struct device *bbram_dev;
132 	uint8_t buf[BUF_ARRAY_CNT];
133 	uint32_t addr;
134 	size_t size = 0;
135 	int ret;
136 
137 	ret = parse_device(sh, argc, argv, &bbram_dev);
138 	if (ret) {
139 		return ret;
140 	}
141 
142 	if (argc < 3) {
143 		shell_error(sh, "Missing address");
144 		return -EINVAL;
145 	}
146 
147 	/* Parse address. */
148 	ret = parse_u32(argv[2], &addr);
149 	if (ret) {
150 		return ret;
151 	}
152 
153 	/* Parse bytes and place them in the buffer. */
154 	for (int i = 3; i < argc; i++) {
155 		ret = parse_u8(argv[i], &buf[i - 3]);
156 		if (ret) {
157 			return ret;
158 		}
159 		size++;
160 	}
161 
162 	if (size == 0) {
163 		shell_error(sh, "Missing data");
164 		return -EINVAL;
165 	}
166 
167 	ret = bbram_write(bbram_dev, addr, size, buf);
168 	if (ret < 0) {
169 		shell_error(sh, "BBRAM write error: %d", ret);
170 		return -EIO;
171 	}
172 
173 	return 0;
174 }
175 
device_name_get(size_t idx,struct shell_static_entry * entry)176 static void device_name_get(size_t idx, struct shell_static_entry *entry)
177 {
178 	const struct device *dev = shell_device_lookup(idx, NULL);
179 
180 	entry->syntax = (dev != NULL) ? dev->name : NULL;
181 	entry->handler = NULL;
182 	entry->help = NULL;
183 	entry->subcmd = NULL;
184 }
185 
186 SHELL_DYNAMIC_CMD_CREATE(dsub_device_name, device_name_get);
187 
188 SHELL_STATIC_SUBCMD_SET_CREATE(bbram_cmds,
189 			       SHELL_CMD_ARG(read, &dsub_device_name,
190 					     "<device> [<address>] [<count>]", cmd_read, 2, 2),
191 			       SHELL_CMD_ARG(write, &dsub_device_name,
192 					     "<device> <address> <byte> [<byte>...]", cmd_write, 4,
193 					     BUF_ARRAY_CNT),
194 			       SHELL_SUBCMD_SET_END);
195 
cmd_bbram(const struct shell * sh,size_t argc,char ** argv)196 static int cmd_bbram(const struct shell *sh, size_t argc, char **argv)
197 {
198 	shell_error(sh, "%s: unknown parameter: %s", argv[0], argv[1]);
199 	return -EINVAL;
200 }
201 
202 SHELL_CMD_ARG_REGISTER(bbram, &bbram_cmds, "Battery-backed RAM shell commands", cmd_bbram, 2, 0);
203