|
@@ -16,6 +16,7 @@
|
|
|
|
|
|
#include <common.h>
|
|
#include <common.h>
|
|
#include <command.h>
|
|
#include <command.h>
|
|
|
|
+#include <dm.h>
|
|
#include <i2c.h>
|
|
#include <i2c.h>
|
|
#include <cros_ec.h>
|
|
#include <cros_ec.h>
|
|
#include <fdtdec.h>
|
|
#include <fdtdec.h>
|
|
@@ -24,6 +25,8 @@
|
|
#include <asm/errno.h>
|
|
#include <asm/errno.h>
|
|
#include <asm/io.h>
|
|
#include <asm/io.h>
|
|
#include <asm-generic/gpio.h>
|
|
#include <asm-generic/gpio.h>
|
|
|
|
+#include <dm/device-internal.h>
|
|
|
|
+#include <dm/uclass-internal.h>
|
|
|
|
|
|
#ifdef DEBUG_TRACE
|
|
#ifdef DEBUG_TRACE
|
|
#define debug_trace(fmt, b...) debug(fmt, #b)
|
|
#define debug_trace(fmt, b...) debug(fmt, #b)
|
|
@@ -38,7 +41,9 @@ enum {
|
|
CROS_EC_CMD_HASH_TIMEOUT_MS = 2000,
|
|
CROS_EC_CMD_HASH_TIMEOUT_MS = 2000,
|
|
};
|
|
};
|
|
|
|
|
|
|
|
+#ifndef CONFIG_DM_CROS_EC
|
|
static struct cros_ec_dev static_dev, *last_dev;
|
|
static struct cros_ec_dev static_dev, *last_dev;
|
|
|
|
+#endif
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
|
@@ -204,6 +209,9 @@ static int send_command_proto3(struct cros_ec_dev *dev,
|
|
const void *dout, int dout_len,
|
|
const void *dout, int dout_len,
|
|
uint8_t **dinp, int din_len)
|
|
uint8_t **dinp, int din_len)
|
|
{
|
|
{
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ struct dm_cros_ec_ops *ops;
|
|
|
|
+#endif
|
|
int out_bytes, in_bytes;
|
|
int out_bytes, in_bytes;
|
|
int rv;
|
|
int rv;
|
|
|
|
|
|
@@ -218,6 +226,10 @@ static int send_command_proto3(struct cros_ec_dev *dev,
|
|
if (in_bytes < 0)
|
|
if (in_bytes < 0)
|
|
return in_bytes;
|
|
return in_bytes;
|
|
|
|
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ ops = dm_cros_ec_get_ops(dev->dev);
|
|
|
|
+ rv = ops->packet(dev->dev, out_bytes, in_bytes);
|
|
|
|
+#else
|
|
switch (dev->interface) {
|
|
switch (dev->interface) {
|
|
#ifdef CONFIG_CROS_EC_SPI
|
|
#ifdef CONFIG_CROS_EC_SPI
|
|
case CROS_EC_IF_SPI:
|
|
case CROS_EC_IF_SPI:
|
|
@@ -235,6 +247,7 @@ static int send_command_proto3(struct cros_ec_dev *dev,
|
|
debug("%s: Unsupported interface\n", __func__);
|
|
debug("%s: Unsupported interface\n", __func__);
|
|
rv = -1;
|
|
rv = -1;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
if (rv < 0)
|
|
if (rv < 0)
|
|
return rv;
|
|
return rv;
|
|
|
|
|
|
@@ -246,6 +259,9 @@ static int send_command(struct cros_ec_dev *dev, uint8_t cmd, int cmd_version,
|
|
const void *dout, int dout_len,
|
|
const void *dout, int dout_len,
|
|
uint8_t **dinp, int din_len)
|
|
uint8_t **dinp, int din_len)
|
|
{
|
|
{
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ struct dm_cros_ec_ops *ops;
|
|
|
|
+#endif
|
|
int ret = -1;
|
|
int ret = -1;
|
|
|
|
|
|
/* Handle protocol version 3 support */
|
|
/* Handle protocol version 3 support */
|
|
@@ -254,6 +270,11 @@ static int send_command(struct cros_ec_dev *dev, uint8_t cmd, int cmd_version,
|
|
dout, dout_len, dinp, din_len);
|
|
dout, dout_len, dinp, din_len);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ ops = dm_cros_ec_get_ops(dev->dev);
|
|
|
|
+ ret = ops->command(dev->dev, cmd, cmd_version,
|
|
|
|
+ (const uint8_t *)dout, dout_len, dinp, din_len);
|
|
|
|
+#else
|
|
switch (dev->interface) {
|
|
switch (dev->interface) {
|
|
#ifdef CONFIG_CROS_EC_SPI
|
|
#ifdef CONFIG_CROS_EC_SPI
|
|
case CROS_EC_IF_SPI:
|
|
case CROS_EC_IF_SPI:
|
|
@@ -280,6 +301,7 @@ static int send_command(struct cros_ec_dev *dev, uint8_t cmd, int cmd_version,
|
|
default:
|
|
default:
|
|
ret = -1;
|
|
ret = -1;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
|
|
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
@@ -990,6 +1012,7 @@ int cros_ec_get_ldo(struct cros_ec_dev *dev, uint8_t index, uint8_t *state)
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+#ifndef CONFIG_DM_CROS_EC
|
|
/**
|
|
/**
|
|
* Decode EC interface details from the device tree and allocate a suitable
|
|
* Decode EC interface details from the device tree and allocate a suitable
|
|
* device.
|
|
* device.
|
|
@@ -1055,11 +1078,61 @@ static int cros_ec_decode_fdt(const void *blob, int node,
|
|
|
|
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
|
|
|
|
-int cros_ec_init(const void *blob, struct cros_ec_dev **cros_ecp)
|
|
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+int cros_ec_register(struct udevice *dev)
|
|
{
|
|
{
|
|
|
|
+ struct cros_ec_dev *cdev = dev->uclass_priv;
|
|
|
|
+ const void *blob = gd->fdt_blob;
|
|
|
|
+ int node = dev->of_offset;
|
|
char id[MSG_BYTES];
|
|
char id[MSG_BYTES];
|
|
|
|
+
|
|
|
|
+ cdev->dev = dev;
|
|
|
|
+ fdtdec_decode_gpio(blob, node, "ec-interrupt", &cdev->ec_int);
|
|
|
|
+ cdev->optimise_flash_write = fdtdec_get_bool(blob, node,
|
|
|
|
+ "optimise-flash-write");
|
|
|
|
+
|
|
|
|
+ /* we will poll the EC interrupt line */
|
|
|
|
+ fdtdec_setup_gpio(&cdev->ec_int);
|
|
|
|
+ if (fdt_gpio_isvalid(&cdev->ec_int)) {
|
|
|
|
+ gpio_request(cdev->ec_int.gpio, "cros-ec-irq");
|
|
|
|
+ gpio_direction_input(cdev->ec_int.gpio);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (cros_ec_check_version(cdev)) {
|
|
|
|
+ debug("%s: Could not detect CROS-EC version\n", __func__);
|
|
|
|
+ return -CROS_EC_ERR_CHECK_VERSION;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (cros_ec_read_id(cdev, id, sizeof(id))) {
|
|
|
|
+ debug("%s: Could not read KBC ID\n", __func__);
|
|
|
|
+ return -CROS_EC_ERR_READ_ID;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Remember this device for use by the cros_ec command */
|
|
|
|
+ debug("Google Chrome EC CROS-EC driver ready, id '%s'\n", id);
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+#else
|
|
|
|
+int cros_ec_init(const void *blob, struct cros_ec_dev **cros_ecp)
|
|
|
|
+{
|
|
struct cros_ec_dev *dev;
|
|
struct cros_ec_dev *dev;
|
|
|
|
+ char id[MSG_BYTES];
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ struct udevice *udev;
|
|
|
|
+ int ret;
|
|
|
|
+
|
|
|
|
+ ret = uclass_find_device(UCLASS_CROS_EC, 0, &udev);
|
|
|
|
+ if (!ret)
|
|
|
|
+ device_remove(udev);
|
|
|
|
+ ret = uclass_get_device(UCLASS_CROS_EC, 0, &udev);
|
|
|
|
+ if (ret)
|
|
|
|
+ return ret;
|
|
|
|
+ dev = udev->uclass_priv;
|
|
|
|
+ return 0;
|
|
|
|
+#else
|
|
int node = 0;
|
|
int node = 0;
|
|
|
|
|
|
*cros_ecp = NULL;
|
|
*cros_ecp = NULL;
|
|
@@ -1108,11 +1181,14 @@ int cros_ec_init(const void *blob, struct cros_ec_dev **cros_ecp)
|
|
default:
|
|
default:
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
|
|
|
|
/* we will poll the EC interrupt line */
|
|
/* we will poll the EC interrupt line */
|
|
fdtdec_setup_gpio(&dev->ec_int);
|
|
fdtdec_setup_gpio(&dev->ec_int);
|
|
- if (fdt_gpio_isvalid(&dev->ec_int))
|
|
|
|
|
|
+ if (fdt_gpio_isvalid(&dev->ec_int)) {
|
|
|
|
+ gpio_request(dev->ec_int.gpio, "cros-ec-irq");
|
|
gpio_direction_input(dev->ec_int.gpio);
|
|
gpio_direction_input(dev->ec_int.gpio);
|
|
|
|
+ }
|
|
|
|
|
|
if (cros_ec_check_version(dev)) {
|
|
if (cros_ec_check_version(dev)) {
|
|
debug("%s: Could not detect CROS-EC version\n", __func__);
|
|
debug("%s: Could not detect CROS-EC version\n", __func__);
|
|
@@ -1125,11 +1201,15 @@ int cros_ec_init(const void *blob, struct cros_ec_dev **cros_ecp)
|
|
}
|
|
}
|
|
|
|
|
|
/* Remember this device for use by the cros_ec command */
|
|
/* Remember this device for use by the cros_ec command */
|
|
- last_dev = *cros_ecp = dev;
|
|
|
|
|
|
+ *cros_ecp = dev;
|
|
|
|
+#ifndef CONFIG_DM_CROS_EC
|
|
|
|
+ last_dev = dev;
|
|
|
|
+#endif
|
|
debug("Google Chrome EC CROS-EC driver ready, id '%s'\n", id);
|
|
debug("Google Chrome EC CROS-EC driver ready, id '%s'\n", id);
|
|
|
|
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
|
|
|
|
int cros_ec_decode_region(int argc, char * const argv[])
|
|
int cros_ec_decode_region(int argc, char * const argv[])
|
|
{
|
|
{
|
|
@@ -1147,15 +1227,10 @@ int cros_ec_decode_region(int argc, char * const argv[])
|
|
return -1;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
-int cros_ec_decode_ec_flash(const void *blob, struct fdt_cros_ec *config)
|
|
|
|
|
|
+int cros_ec_decode_ec_flash(const void *blob, int node,
|
|
|
|
+ struct fdt_cros_ec *config)
|
|
{
|
|
{
|
|
- int flash_node, node;
|
|
|
|
-
|
|
|
|
- node = fdtdec_next_compatible(blob, 0, COMPAT_GOOGLE_CROS_EC);
|
|
|
|
- if (node < 0) {
|
|
|
|
- debug("Failed to find chrome-ec node'\n");
|
|
|
|
- return -1;
|
|
|
|
- }
|
|
|
|
|
|
+ int flash_node;
|
|
|
|
|
|
flash_node = fdt_subnode_offset(blob, node, "flash");
|
|
flash_node = fdt_subnode_offset(blob, node, "flash");
|
|
if (flash_node < 0) {
|
|
if (flash_node < 0) {
|
|
@@ -1516,7 +1591,10 @@ static int cros_ec_i2c_passthrough(struct cros_ec_dev *dev, int flag,
|
|
|
|
|
|
static int do_cros_ec(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
|
static int do_cros_ec(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
|
{
|
|
{
|
|
- struct cros_ec_dev *dev = last_dev;
|
|
|
|
|
|
+ struct cros_ec_dev *dev;
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ struct udevice *udev;
|
|
|
|
+#endif
|
|
const char *cmd;
|
|
const char *cmd;
|
|
int ret = 0;
|
|
int ret = 0;
|
|
|
|
|
|
@@ -1525,19 +1603,31 @@ static int do_cros_ec(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
|
|
|
|
|
|
cmd = argv[1];
|
|
cmd = argv[1];
|
|
if (0 == strcmp("init", cmd)) {
|
|
if (0 == strcmp("init", cmd)) {
|
|
|
|
+#ifndef CONFIG_DM_CROS_EC
|
|
ret = cros_ec_init(gd->fdt_blob, &dev);
|
|
ret = cros_ec_init(gd->fdt_blob, &dev);
|
|
if (ret) {
|
|
if (ret) {
|
|
printf("Could not init cros_ec device (err %d)\n", ret);
|
|
printf("Could not init cros_ec device (err %d)\n", ret);
|
|
return 1;
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
+#endif
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+ ret = uclass_get_device(UCLASS_CROS_EC, 0, &udev);
|
|
|
|
+ if (ret) {
|
|
|
|
+ printf("Cannot get cros-ec device (err=%d)\n", ret);
|
|
|
|
+ return 1;
|
|
|
|
+ }
|
|
|
|
+ dev = udev->uclass_priv;
|
|
|
|
+#else
|
|
/* Just use the last allocated device; there should be only one */
|
|
/* Just use the last allocated device; there should be only one */
|
|
if (!last_dev) {
|
|
if (!last_dev) {
|
|
printf("No CROS-EC device available\n");
|
|
printf("No CROS-EC device available\n");
|
|
return 1;
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
+ dev = last_dev;
|
|
|
|
+#endif
|
|
if (0 == strcmp("id", cmd)) {
|
|
if (0 == strcmp("id", cmd)) {
|
|
char id[MSG_BYTES];
|
|
char id[MSG_BYTES];
|
|
|
|
|
|
@@ -1794,3 +1884,11 @@ U_BOOT_CMD(
|
|
"crosec i2c mw chip address[.0, .1, .2] value [count] - write to I2C passthru (fill)"
|
|
"crosec i2c mw chip address[.0, .1, .2] value [count] - write to I2C passthru (fill)"
|
|
);
|
|
);
|
|
#endif
|
|
#endif
|
|
|
|
+
|
|
|
|
+#ifdef CONFIG_DM_CROS_EC
|
|
|
|
+UCLASS_DRIVER(cros_ec) = {
|
|
|
|
+ .id = UCLASS_CROS_EC,
|
|
|
|
+ .name = "cros_ec",
|
|
|
|
+ .per_device_auto_alloc_size = sizeof(struct cros_ec_dev),
|
|
|
|
+};
|
|
|
|
+#endif
|