xref: /OK3568_Linux_fs/u-boot/board/Synology/ds414/cmd_syno.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Commands to deal with Synology specifics.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Copyright (C) 2015  Phil Sutter <phil@nwl.cc>
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  * SPDX-License-Identifier:	GPL-2.0+
7*4882a593Smuzhiyun  */
8*4882a593Smuzhiyun 
9*4882a593Smuzhiyun #include <common.h>
10*4882a593Smuzhiyun #include <div64.h>
11*4882a593Smuzhiyun #include <spi.h>
12*4882a593Smuzhiyun #include <spi_flash.h>
13*4882a593Smuzhiyun #include <linux/mtd/mtd.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #include <asm/io.h>
16*4882a593Smuzhiyun #include "../drivers/ddr/marvell/axp/ddr3_init.h"
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun #define ETH_ALEN		6
19*4882a593Smuzhiyun #define ETHADDR_MAX		4
20*4882a593Smuzhiyun #define SYNO_SN_TAG		"SN="
21*4882a593Smuzhiyun #define SYNO_CHKSUM_TAG		"CHK="
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun 
do_syno_populate(int argc,char * const argv[])24*4882a593Smuzhiyun static int do_syno_populate(int argc, char * const argv[])
25*4882a593Smuzhiyun {
26*4882a593Smuzhiyun 	unsigned int bus = CONFIG_SF_DEFAULT_BUS;
27*4882a593Smuzhiyun 	unsigned int cs = CONFIG_SF_DEFAULT_CS;
28*4882a593Smuzhiyun 	unsigned int speed = CONFIG_SF_DEFAULT_SPEED;
29*4882a593Smuzhiyun 	unsigned int mode = CONFIG_SF_DEFAULT_MODE;
30*4882a593Smuzhiyun 	struct spi_flash *flash;
31*4882a593Smuzhiyun 	unsigned long addr = 0x80000; /* XXX: parameterize this? */
32*4882a593Smuzhiyun 	loff_t offset = 0x007d0000;
33*4882a593Smuzhiyun 	loff_t len = 0x00010000;
34*4882a593Smuzhiyun 	char *buf, *bufp;
35*4882a593Smuzhiyun 	char var[128];
36*4882a593Smuzhiyun 	char val[128];
37*4882a593Smuzhiyun 	int ret, n;
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun 	/* XXX: arg parsing to select flash here? */
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun 	flash = spi_flash_probe(bus, cs, speed, mode);
42*4882a593Smuzhiyun 	if (!flash) {
43*4882a593Smuzhiyun 		printf("Failed to initialize SPI flash at %u:%u\n", bus, cs);
44*4882a593Smuzhiyun 		return 1;
45*4882a593Smuzhiyun 	}
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	buf = map_physmem(addr, len, MAP_WRBACK);
48*4882a593Smuzhiyun 	if (!buf) {
49*4882a593Smuzhiyun 		puts("Failed to map physical memory\n");
50*4882a593Smuzhiyun 		return 1;
51*4882a593Smuzhiyun 	}
52*4882a593Smuzhiyun 
53*4882a593Smuzhiyun 	ret = spi_flash_read(flash, offset, len, buf);
54*4882a593Smuzhiyun 	if (ret) {
55*4882a593Smuzhiyun 		puts("Failed to read from SPI flash\n");
56*4882a593Smuzhiyun 		goto out_unmap;
57*4882a593Smuzhiyun 	}
58*4882a593Smuzhiyun 
59*4882a593Smuzhiyun 	for (n = 0; n < ETHADDR_MAX; n++) {
60*4882a593Smuzhiyun 		char ethaddr[ETH_ALEN];
61*4882a593Smuzhiyun 		int i, sum = 0;
62*4882a593Smuzhiyun 		unsigned char csum = 0;
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun 		for (i = 0, bufp = buf + n * 7; i < ETH_ALEN; i++) {
65*4882a593Smuzhiyun 			sum += bufp[i];
66*4882a593Smuzhiyun 			csum += bufp[i];
67*4882a593Smuzhiyun 			ethaddr[i] = bufp[i];
68*4882a593Smuzhiyun 		}
69*4882a593Smuzhiyun 		if (!sum)		/* MAC address empty */
70*4882a593Smuzhiyun 			continue;
71*4882a593Smuzhiyun 		if (csum != bufp[i]) {	/* seventh byte is checksum value */
72*4882a593Smuzhiyun 			printf("Invalid MAC address for interface %d!\n", n);
73*4882a593Smuzhiyun 			continue;
74*4882a593Smuzhiyun 		}
75*4882a593Smuzhiyun 		if (n == 0)
76*4882a593Smuzhiyun 			sprintf(var, "ethaddr");
77*4882a593Smuzhiyun 		else
78*4882a593Smuzhiyun 			sprintf(var, "eth%daddr", n);
79*4882a593Smuzhiyun 		snprintf(val, sizeof(val) - 1,
80*4882a593Smuzhiyun 		         "%02x:%02x:%02x:%02x:%02x:%02x",
81*4882a593Smuzhiyun 		         ethaddr[0], ethaddr[1], ethaddr[2],
82*4882a593Smuzhiyun 			 ethaddr[3], ethaddr[4], ethaddr[5]);
83*4882a593Smuzhiyun 		printf("parsed %s = %s\n", var, val);
84*4882a593Smuzhiyun 		env_set(var, val);
85*4882a593Smuzhiyun 	}
86*4882a593Smuzhiyun 	if (!strncmp(buf + 32, SYNO_SN_TAG, strlen(SYNO_SN_TAG))) {
87*4882a593Smuzhiyun 		char *snp, *csump;
88*4882a593Smuzhiyun 		int csum = 0;
89*4882a593Smuzhiyun 		unsigned long c;
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun 		snp = bufp = buf + 32 + strlen(SYNO_SN_TAG);
92*4882a593Smuzhiyun 		for (n = 0; bufp[n] && bufp[n] != ','; n++)
93*4882a593Smuzhiyun 			csum += bufp[n];
94*4882a593Smuzhiyun 		bufp[n] = '\0';
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun 		/* should come right after, but you never know */
97*4882a593Smuzhiyun 		bufp = strstr(bufp + n + 1, SYNO_CHKSUM_TAG);
98*4882a593Smuzhiyun 		if (!bufp) {
99*4882a593Smuzhiyun 			printf("Serial number checksum tag missing!\n");
100*4882a593Smuzhiyun 			goto out_unmap;
101*4882a593Smuzhiyun 		}
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun 		csump = bufp += strlen(SYNO_CHKSUM_TAG);
104*4882a593Smuzhiyun 		for (n = 0; bufp[n] && bufp[n] != ','; n++)
105*4882a593Smuzhiyun 			;
106*4882a593Smuzhiyun 		bufp[n] = '\0';
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 		if (strict_strtoul(csump, 10, &c) || c != csum) {
109*4882a593Smuzhiyun 			puts("Invalid serial number found!\n");
110*4882a593Smuzhiyun 			ret = 1;
111*4882a593Smuzhiyun 			goto out_unmap;
112*4882a593Smuzhiyun 		}
113*4882a593Smuzhiyun 		printf("parsed SN = %s\n", snp);
114*4882a593Smuzhiyun 		env_set("SN", snp);
115*4882a593Smuzhiyun 	} else {	/* old style format */
116*4882a593Smuzhiyun 		unsigned char csum = 0;
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 		for (n = 0, bufp = buf + 32; n < 10; n++)
119*4882a593Smuzhiyun 			csum += bufp[n];
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun 		if (csum != bufp[n]) {
122*4882a593Smuzhiyun 			puts("Invalid serial number found!\n");
123*4882a593Smuzhiyun 			ret = 1;
124*4882a593Smuzhiyun 			goto out_unmap;
125*4882a593Smuzhiyun 		}
126*4882a593Smuzhiyun 		bufp[n] = '\0';
127*4882a593Smuzhiyun 		printf("parsed SN = %s\n", buf + 32);
128*4882a593Smuzhiyun 		env_set("SN", buf + 32);
129*4882a593Smuzhiyun 	}
130*4882a593Smuzhiyun out_unmap:
131*4882a593Smuzhiyun 	unmap_physmem(buf, len);
132*4882a593Smuzhiyun 	return ret;
133*4882a593Smuzhiyun }
134*4882a593Smuzhiyun 
135*4882a593Smuzhiyun /* map bit position to function in POWER_MNG_CTRL_REG */
136*4882a593Smuzhiyun static const char * const pwr_mng_bit_func[] = {
137*4882a593Smuzhiyun 	"audio",
138*4882a593Smuzhiyun 	"ge3", "ge2", "ge1", "ge0",
139*4882a593Smuzhiyun 	"pcie00", "pcie01", "pcie02", "pcie03",
140*4882a593Smuzhiyun 	"pcie10", "pcie11", "pcie12", "pcie13",
141*4882a593Smuzhiyun 	"bp",
142*4882a593Smuzhiyun 	"sata0_link", "sata0_core",
143*4882a593Smuzhiyun 	"lcd",
144*4882a593Smuzhiyun 	"sdio",
145*4882a593Smuzhiyun 	"usb0", "usb1", "usb2",
146*4882a593Smuzhiyun 	"idma", "xor0", "crypto",
147*4882a593Smuzhiyun 	NULL,
148*4882a593Smuzhiyun 	"tdm",
149*4882a593Smuzhiyun 	"pcie20", "pcie30",
150*4882a593Smuzhiyun 	"xor1",
151*4882a593Smuzhiyun 	"sata1_link", "sata1_core",
152*4882a593Smuzhiyun 	NULL,
153*4882a593Smuzhiyun };
154*4882a593Smuzhiyun 
do_syno_clk_gate(int argc,char * const argv[])155*4882a593Smuzhiyun static int do_syno_clk_gate(int argc, char * const argv[])
156*4882a593Smuzhiyun {
157*4882a593Smuzhiyun 	u32 pwr_mng_ctrl_reg = reg_read(POWER_MNG_CTRL_REG);
158*4882a593Smuzhiyun 	const char *func, *state;
159*4882a593Smuzhiyun 	int i, val;
160*4882a593Smuzhiyun 
161*4882a593Smuzhiyun 	if (argc < 2)
162*4882a593Smuzhiyun 		return -1;
163*4882a593Smuzhiyun 
164*4882a593Smuzhiyun 	if (!strcmp(argv[1], "get")) {
165*4882a593Smuzhiyun 		puts("Clock Gating:\n");
166*4882a593Smuzhiyun 		for (i = 0; i < 32; i++) {
167*4882a593Smuzhiyun 			func = pwr_mng_bit_func[i];
168*4882a593Smuzhiyun 			if (!func)
169*4882a593Smuzhiyun 				continue;
170*4882a593Smuzhiyun 			state = pwr_mng_ctrl_reg & (1 << i) ?  "ON" : "OFF";
171*4882a593Smuzhiyun 			printf("%s:\t\t%s\n", func, state);
172*4882a593Smuzhiyun 		}
173*4882a593Smuzhiyun 		return 0;
174*4882a593Smuzhiyun 	}
175*4882a593Smuzhiyun 	if (argc < 4)
176*4882a593Smuzhiyun 		return -1;
177*4882a593Smuzhiyun 	if (!strcmp(argv[1], "set")) {
178*4882a593Smuzhiyun 		func = argv[2];
179*4882a593Smuzhiyun 		state = argv[3];
180*4882a593Smuzhiyun 		for (i = 0; i < 32; i++) {
181*4882a593Smuzhiyun 			if (!pwr_mng_bit_func[i])
182*4882a593Smuzhiyun 				continue;
183*4882a593Smuzhiyun 			if (!strcmp(func, pwr_mng_bit_func[i]))
184*4882a593Smuzhiyun 				break;
185*4882a593Smuzhiyun 		}
186*4882a593Smuzhiyun 		if (i == 32) {
187*4882a593Smuzhiyun 			printf("Error: name '%s' not known\n", func);
188*4882a593Smuzhiyun 			return -1;
189*4882a593Smuzhiyun 		}
190*4882a593Smuzhiyun 		val = state[0] != '0';
191*4882a593Smuzhiyun 		pwr_mng_ctrl_reg |= (val << i);
192*4882a593Smuzhiyun 		pwr_mng_ctrl_reg &= ~(!val << i);
193*4882a593Smuzhiyun 		reg_write(POWER_MNG_CTRL_REG, pwr_mng_ctrl_reg);
194*4882a593Smuzhiyun 	}
195*4882a593Smuzhiyun 	return 0;
196*4882a593Smuzhiyun }
197*4882a593Smuzhiyun 
do_syno(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])198*4882a593Smuzhiyun static int do_syno(cmd_tbl_t *cmdtp, int flag,
199*4882a593Smuzhiyun                    int argc, char * const argv[])
200*4882a593Smuzhiyun {
201*4882a593Smuzhiyun 	const char *cmd;
202*4882a593Smuzhiyun 	int ret = 0;
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun 	if (argc < 2)
205*4882a593Smuzhiyun 		goto usage;
206*4882a593Smuzhiyun 
207*4882a593Smuzhiyun 	cmd = argv[1];
208*4882a593Smuzhiyun 	--argc;
209*4882a593Smuzhiyun 	++argv;
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun 	if (!strcmp(cmd, "populate_env"))
212*4882a593Smuzhiyun 		ret = do_syno_populate(argc, argv);
213*4882a593Smuzhiyun 	else if (!strcmp(cmd, "clk_gate"))
214*4882a593Smuzhiyun 		ret = do_syno_clk_gate(argc, argv);
215*4882a593Smuzhiyun 
216*4882a593Smuzhiyun 	if (ret != -1)
217*4882a593Smuzhiyun 		return ret;
218*4882a593Smuzhiyun usage:
219*4882a593Smuzhiyun 	return CMD_RET_USAGE;
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun U_BOOT_CMD(
223*4882a593Smuzhiyun 	syno, 5, 1, do_syno,
224*4882a593Smuzhiyun 	"Synology specific commands",
225*4882a593Smuzhiyun 	"populate_env                 - Read vendor data from SPI flash into environment\n"
226*4882a593Smuzhiyun 	"clk_gate (get|set name 1|0)  - Manage clock gating\n"
227*4882a593Smuzhiyun );
228