ee5931d488
This adds information about the Gateworks System Controller to the gsc command such as the firmware version, firmware CRC and status of the GSC watchdog (if its enabled and if its tripped). Additionally the 'gsc wd' command can be used to enable or disable the watchdog with the following usage: gsc wd enable [30|60] gsc wd disable Note that the GSC registers are battery-backed by the GSC coincell so once eanbled, they remain enabled across power-cycles or until either the GSC firmware has been updated or FLASH has been re-programmed by the Gateworks JTAG adapter. Signed-off-by: Tim Harvey <tharvey@gateworks.com>
192 lines
4.7 KiB
C
192 lines
4.7 KiB
C
/*
|
|
* Copyright (C) 2013 Gateworks Corporation
|
|
*
|
|
* Author: Tim Harvey <tharvey@gateworks.com>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <asm/errno.h>
|
|
#include <common.h>
|
|
#include <i2c.h>
|
|
#include <linux/ctype.h>
|
|
|
|
#include "gsc.h"
|
|
|
|
/*
|
|
* The Gateworks System Controller will fail to ACK a master transaction if
|
|
* it is busy, which can occur during its 1HZ timer tick while reading ADC's.
|
|
* When this does occur, it will never be busy long enough to fail more than
|
|
* 2 back-to-back transfers. Thus we wrap i2c_read and i2c_write with
|
|
* 3 retries.
|
|
*/
|
|
int gsc_i2c_read(uchar chip, uint addr, int alen, uchar *buf, int len)
|
|
{
|
|
int retry = 3;
|
|
int n = 0;
|
|
int ret;
|
|
|
|
while (n++ < retry) {
|
|
ret = i2c_read(chip, addr, alen, buf, len);
|
|
if (!ret)
|
|
break;
|
|
debug("%s: 0x%02x 0x%02x retry%d: %d\n", __func__, chip, addr,
|
|
n, ret);
|
|
if (ret != -ENODEV)
|
|
break;
|
|
mdelay(10);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int gsc_i2c_write(uchar chip, uint addr, int alen, uchar *buf, int len)
|
|
{
|
|
int retry = 3;
|
|
int n = 0;
|
|
int ret;
|
|
|
|
while (n++ < retry) {
|
|
ret = i2c_write(chip, addr, alen, buf, len);
|
|
if (!ret)
|
|
break;
|
|
debug("%s: 0x%02x 0x%02x retry%d: %d\n", __func__, chip, addr,
|
|
n, ret);
|
|
if (ret != -ENODEV)
|
|
break;
|
|
mdelay(10);
|
|
}
|
|
mdelay(100);
|
|
return ret;
|
|
}
|
|
|
|
static void read_hwmon(const char *name, uint reg, uint size)
|
|
{
|
|
unsigned char buf[3];
|
|
uint ui;
|
|
|
|
printf("%-8s:", name);
|
|
memset(buf, 0, sizeof(buf));
|
|
if (gsc_i2c_read(GSC_HWMON_ADDR, reg, 1, buf, size)) {
|
|
puts("fRD\n");
|
|
} else {
|
|
ui = buf[0] | (buf[1]<<8) | (buf[2]<<16);
|
|
if (ui == 0xffffff)
|
|
puts("invalid\n");
|
|
else
|
|
printf("%d\n", ui);
|
|
}
|
|
}
|
|
|
|
int gsc_info(int verbose)
|
|
{
|
|
const char *model = getenv("model");
|
|
unsigned char buf[16];
|
|
|
|
i2c_set_bus_num(0);
|
|
if (gsc_i2c_read(GSC_SC_ADDR, 0, 1, buf, 16))
|
|
return CMD_RET_FAILURE;
|
|
|
|
printf("GSC: v%d", buf[GSC_SC_FWVER]);
|
|
printf(" 0x%04x", buf[GSC_SC_FWCRC] | buf[GSC_SC_FWCRC+1]<<8);
|
|
printf(" WDT:%sabled", (buf[GSC_SC_CTRL1] & (1<<GSC_SC_CTRL1_WDEN))
|
|
? "en" : "dis");
|
|
if (buf[GSC_SC_STATUS] & (1 << GSC_SC_IRQ_WATCHDOG)) {
|
|
buf[GSC_SC_STATUS] &= ~(1 << GSC_SC_IRQ_WATCHDOG);
|
|
puts(" WDT_RESET");
|
|
gsc_i2c_write(GSC_SC_ADDR, GSC_SC_STATUS, 1,
|
|
&buf[GSC_SC_STATUS], 1);
|
|
}
|
|
puts("\n");
|
|
if (!verbose)
|
|
return CMD_RET_SUCCESS;
|
|
|
|
read_hwmon("Temp", GSC_HWMON_TEMP, 2);
|
|
read_hwmon("VIN", GSC_HWMON_VIN, 3);
|
|
read_hwmon("VBATT", GSC_HWMON_VBATT, 3);
|
|
read_hwmon("VDD_3P3", GSC_HWMON_VDD_3P3, 3);
|
|
read_hwmon("VDD_ARM", GSC_HWMON_VDD_CORE, 3);
|
|
read_hwmon("VDD_SOC", GSC_HWMON_VDD_SOC, 3);
|
|
read_hwmon("VDD_HIGH", GSC_HWMON_VDD_HIGH, 3);
|
|
read_hwmon("VDD_DDR", GSC_HWMON_VDD_DDR, 3);
|
|
read_hwmon("VDD_5P0", GSC_HWMON_VDD_5P0, 3);
|
|
read_hwmon("VDD_2P5", GSC_HWMON_VDD_2P5, 3);
|
|
read_hwmon("VDD_1P8", GSC_HWMON_VDD_1P8, 3);
|
|
read_hwmon("VDD_IO2", GSC_HWMON_VDD_IO2, 3);
|
|
switch (model[3]) {
|
|
case '1': /* GW51xx */
|
|
read_hwmon("VDD_IO3", GSC_HWMON_VDD_IO4, 3); /* -C rev */
|
|
break;
|
|
case '2': /* GW52xx */
|
|
break;
|
|
case '3': /* GW53xx */
|
|
read_hwmon("VDD_IO4", GSC_HWMON_VDD_IO4, 3); /* -C rev */
|
|
read_hwmon("VDD_GPS", GSC_HWMON_VDD_IO3, 3);
|
|
break;
|
|
case '4': /* GW54xx */
|
|
read_hwmon("VDD_IO3", GSC_HWMON_VDD_IO4, 3); /* -C rev */
|
|
read_hwmon("VDD_GPS", GSC_HWMON_VDD_IO3, 3);
|
|
break;
|
|
case '5': /* GW55xx */
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_CMD_GSC
|
|
static int do_gsc_wd(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
|
{
|
|
unsigned char reg;
|
|
|
|
if (argc < 2)
|
|
return CMD_RET_USAGE;
|
|
|
|
if (strcasecmp(argv[1], "enable") == 0) {
|
|
int timeout = 0;
|
|
|
|
if (argc > 2)
|
|
timeout = simple_strtoul(argv[2], NULL, 10);
|
|
i2c_set_bus_num(0);
|
|
if (gsc_i2c_read(GSC_SC_ADDR, GSC_SC_CTRL1, 1, ®, 1))
|
|
return CMD_RET_FAILURE;
|
|
reg &= ~((1 << GSC_SC_CTRL1_WDEN) | (1 << GSC_SC_CTRL1_WDTIME));
|
|
if (timeout == 60)
|
|
reg |= (1 << GSC_SC_CTRL1_WDTIME);
|
|
else
|
|
timeout = 30;
|
|
reg |= (1 << GSC_SC_CTRL1_WDEN);
|
|
if (gsc_i2c_write(GSC_SC_ADDR, GSC_SC_CTRL1, 1, ®, 1))
|
|
return CMD_RET_FAILURE;
|
|
printf("GSC Watchdog enabled with timeout=%d seconds\n",
|
|
timeout);
|
|
} else if (strcasecmp(argv[1], "disable") == 0) {
|
|
i2c_set_bus_num(0);
|
|
if (gsc_i2c_read(GSC_SC_ADDR, GSC_SC_CTRL1, 1, ®, 1))
|
|
return CMD_RET_FAILURE;
|
|
reg &= ~((1 << GSC_SC_CTRL1_WDEN) | (1 << GSC_SC_CTRL1_WDTIME));
|
|
if (gsc_i2c_write(GSC_SC_ADDR, GSC_SC_CTRL1, 1, ®, 1))
|
|
return CMD_RET_FAILURE;
|
|
printf("GSC Watchdog disabled\n");
|
|
} else {
|
|
return CMD_RET_USAGE;
|
|
}
|
|
return CMD_RET_SUCCESS;
|
|
}
|
|
|
|
static int do_gsc(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
|
{
|
|
if (argc < 2)
|
|
return gsc_info(1);
|
|
|
|
if (strcasecmp(argv[1], "wd") == 0)
|
|
return do_gsc_wd(cmdtp, flag, --argc, ++argv);
|
|
|
|
return CMD_RET_USAGE;
|
|
}
|
|
|
|
U_BOOT_CMD(
|
|
gsc, 4, 1, do_gsc, "GSC configuration",
|
|
"[wd enable [30|60]]|[wd disable]\n"
|
|
);
|
|
|
|
#endif /* CONFIG_CMD_GSC */
|