commit
79493609c5
@ -0,0 +1,365 @@ |
||||
/*
|
||||
* Chromium OS cros_ec driver |
||||
* |
||||
* Copyright (c) 2016 The Chromium OS Authors. |
||||
* Copyright (c) 2016 National Instruments Corp |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <command.h> |
||||
#include <cros_ec.h> |
||||
#include <dm.h> |
||||
#include <dm/device-internal.h> |
||||
#include <dm/uclass-internal.h> |
||||
|
||||
/* Note: depends on enum ec_current_image */ |
||||
static const char * const ec_current_image_name[] = {"unknown", "RO", "RW"}; |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
/**
|
||||
* Perform a flash read or write command |
||||
* |
||||
* @param dev CROS-EC device to read/write |
||||
* @param is_write 1 do to a write, 0 to do a read |
||||
* @param argc Number of arguments |
||||
* @param argv Arguments (2 is region, 3 is address) |
||||
* @return 0 for ok, 1 for a usage error or -ve for ec command error |
||||
* (negative EC_RES_...) |
||||
*/ |
||||
static int do_read_write(struct cros_ec_dev *dev, int is_write, int argc, |
||||
char * const argv[]) |
||||
{ |
||||
uint32_t offset, size = -1U, region_size; |
||||
unsigned long addr; |
||||
char *endp; |
||||
int region; |
||||
int ret; |
||||
|
||||
region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
if (region == -1) |
||||
return 1; |
||||
if (argc < 4) |
||||
return 1; |
||||
addr = simple_strtoul(argv[3], &endp, 16); |
||||
if (*argv[3] == 0 || *endp != 0) |
||||
return 1; |
||||
if (argc > 4) { |
||||
size = simple_strtoul(argv[4], &endp, 16); |
||||
if (*argv[4] == 0 || *endp != 0) |
||||
return 1; |
||||
} |
||||
|
||||
ret = cros_ec_flash_offset(dev, region, &offset, ®ion_size); |
||||
if (ret) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
return ret; |
||||
} |
||||
if (size == -1U) |
||||
size = region_size; |
||||
|
||||
ret = is_write ? |
||||
cros_ec_flash_write(dev, (uint8_t *)addr, offset, size) : |
||||
cros_ec_flash_read(dev, (uint8_t *)addr, offset, size); |
||||
if (ret) { |
||||
debug("%s: Could not %s region\n", __func__, |
||||
is_write ? "write" : "read"); |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int do_cros_ec(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) |
||||
{ |
||||
struct cros_ec_dev *dev; |
||||
struct udevice *udev; |
||||
const char *cmd; |
||||
int ret = 0; |
||||
|
||||
if (argc < 2) |
||||
return CMD_RET_USAGE; |
||||
|
||||
cmd = argv[1]; |
||||
if (0 == strcmp("init", cmd)) { |
||||
/* Remove any existing device */ |
||||
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) { |
||||
printf("Could not init cros_ec device (err %d)\n", ret); |
||||
return 1; |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
ret = uclass_get_device(UCLASS_CROS_EC, 0, &udev); |
||||
if (ret) { |
||||
printf("Cannot get cros-ec device (err=%d)\n", ret); |
||||
return 1; |
||||
} |
||||
dev = dev_get_uclass_priv(udev); |
||||
if (0 == strcmp("id", cmd)) { |
||||
char id[MSG_BYTES]; |
||||
|
||||
if (cros_ec_read_id(dev, id, sizeof(id))) { |
||||
debug("%s: Could not read KBC ID\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("%s\n", id); |
||||
} else if (0 == strcmp("info", cmd)) { |
||||
struct ec_response_mkbp_info info; |
||||
|
||||
if (cros_ec_info(dev, &info)) { |
||||
debug("%s: Could not read KBC info\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("rows = %u\n", info.rows); |
||||
printf("cols = %u\n", info.cols); |
||||
printf("switches = %#x\n", info.switches); |
||||
} else if (0 == strcmp("curimage", cmd)) { |
||||
enum ec_current_image image; |
||||
|
||||
if (cros_ec_read_current_image(dev, &image)) { |
||||
debug("%s: Could not read KBC image\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("%d\n", image); |
||||
} else if (0 == strcmp("hash", cmd)) { |
||||
struct ec_response_vboot_hash hash; |
||||
int i; |
||||
|
||||
if (cros_ec_read_hash(dev, &hash)) { |
||||
debug("%s: Could not read KBC hash\n", __func__); |
||||
return 1; |
||||
} |
||||
|
||||
if (hash.hash_type == EC_VBOOT_HASH_TYPE_SHA256) |
||||
printf("type: SHA-256\n"); |
||||
else |
||||
printf("type: %d\n", hash.hash_type); |
||||
|
||||
printf("offset: 0x%08x\n", hash.offset); |
||||
printf("size: 0x%08x\n", hash.size); |
||||
|
||||
printf("digest: "); |
||||
for (i = 0; i < hash.digest_size; i++) |
||||
printf("%02x", hash.hash_digest[i]); |
||||
printf("\n"); |
||||
} else if (0 == strcmp("reboot", cmd)) { |
||||
int region; |
||||
enum ec_reboot_cmd cmd; |
||||
|
||||
if (argc >= 3 && !strcmp(argv[2], "cold")) { |
||||
cmd = EC_REBOOT_COLD; |
||||
} else { |
||||
region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
if (region == EC_FLASH_REGION_RO) |
||||
cmd = EC_REBOOT_JUMP_RO; |
||||
else if (region == EC_FLASH_REGION_RW) |
||||
cmd = EC_REBOOT_JUMP_RW; |
||||
else |
||||
return CMD_RET_USAGE; |
||||
} |
||||
|
||||
if (cros_ec_reboot(dev, cmd, 0)) { |
||||
debug("%s: Could not reboot KBC\n", __func__); |
||||
return 1; |
||||
} |
||||
} else if (0 == strcmp("events", cmd)) { |
||||
uint32_t events; |
||||
|
||||
if (cros_ec_get_host_events(dev, &events)) { |
||||
debug("%s: Could not read host events\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("0x%08x\n", events); |
||||
} else if (0 == strcmp("clrevents", cmd)) { |
||||
uint32_t events = 0x7fffffff; |
||||
|
||||
if (argc >= 3) |
||||
events = simple_strtol(argv[2], NULL, 0); |
||||
|
||||
if (cros_ec_clear_host_events(dev, events)) { |
||||
debug("%s: Could not clear host events\n", __func__); |
||||
return 1; |
||||
} |
||||
} else if (0 == strcmp("read", cmd)) { |
||||
ret = do_read_write(dev, 0, argc, argv); |
||||
if (ret > 0) |
||||
return CMD_RET_USAGE; |
||||
} else if (0 == strcmp("write", cmd)) { |
||||
ret = do_read_write(dev, 1, argc, argv); |
||||
if (ret > 0) |
||||
return CMD_RET_USAGE; |
||||
} else if (0 == strcmp("erase", cmd)) { |
||||
int region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
uint32_t offset, size; |
||||
|
||||
if (region == -1) |
||||
return CMD_RET_USAGE; |
||||
if (cros_ec_flash_offset(dev, region, &offset, &size)) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
ret = -1; |
||||
} else { |
||||
ret = cros_ec_flash_erase(dev, offset, size); |
||||
if (ret) { |
||||
debug("%s: Could not erase region\n", |
||||
__func__); |
||||
} |
||||
} |
||||
} else if (0 == strcmp("regioninfo", cmd)) { |
||||
int region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
uint32_t offset, size; |
||||
|
||||
if (region == -1) |
||||
return CMD_RET_USAGE; |
||||
ret = cros_ec_flash_offset(dev, region, &offset, &size); |
||||
if (ret) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
} else { |
||||
printf("Region: %s\n", region == EC_FLASH_REGION_RO ? |
||||
"RO" : "RW"); |
||||
printf("Offset: %x\n", offset); |
||||
printf("Size: %x\n", size); |
||||
} |
||||
} else if (0 == strcmp("flashinfo", cmd)) { |
||||
struct ec_response_flash_info p; |
||||
|
||||
ret = cros_ec_read_flashinfo(dev, &p); |
||||
if (!ret) { |
||||
printf("Flash size: %u\n", p.flash_size); |
||||
printf("Write block size: %u\n", p.write_block_size); |
||||
printf("Erase block size: %u\n", p.erase_block_size); |
||||
} |
||||
} else if (0 == strcmp("vbnvcontext", cmd)) { |
||||
uint8_t block[EC_VBNV_BLOCK_SIZE]; |
||||
char buf[3]; |
||||
int i, len; |
||||
unsigned long result; |
||||
|
||||
if (argc <= 2) { |
||||
ret = cros_ec_read_vbnvcontext(dev, block); |
||||
if (!ret) { |
||||
printf("vbnv_block: "); |
||||
for (i = 0; i < EC_VBNV_BLOCK_SIZE; i++) |
||||
printf("%02x", block[i]); |
||||
putc('\n'); |
||||
} |
||||
} else { |
||||
/*
|
||||
* TODO(clchiou): Move this to a utility function as |
||||
* cmd_spi might want to call it. |
||||
*/ |
||||
memset(block, 0, EC_VBNV_BLOCK_SIZE); |
||||
len = strlen(argv[2]); |
||||
buf[2] = '\0'; |
||||
for (i = 0; i < EC_VBNV_BLOCK_SIZE; i++) { |
||||
if (i * 2 >= len) |
||||
break; |
||||
buf[0] = argv[2][i * 2]; |
||||
if (i * 2 + 1 >= len) |
||||
buf[1] = '0'; |
||||
else |
||||
buf[1] = argv[2][i * 2 + 1]; |
||||
strict_strtoul(buf, 16, &result); |
||||
block[i] = result; |
||||
} |
||||
ret = cros_ec_write_vbnvcontext(dev, block); |
||||
} |
||||
if (ret) { |
||||
debug("%s: Could not %s VbNvContext\n", __func__, |
||||
argc <= 2 ? "read" : "write"); |
||||
} |
||||
} else if (0 == strcmp("test", cmd)) { |
||||
int result = cros_ec_test(dev); |
||||
|
||||
if (result) |
||||
printf("Test failed with error %d\n", result); |
||||
else |
||||
puts("Test passed\n"); |
||||
} else if (0 == strcmp("version", cmd)) { |
||||
struct ec_response_get_version *p; |
||||
char *build_string; |
||||
|
||||
ret = cros_ec_read_version(dev, &p); |
||||
if (!ret) { |
||||
/* Print versions */ |
||||
printf("RO version: %1.*s\n", |
||||
(int)sizeof(p->version_string_ro), |
||||
p->version_string_ro); |
||||
printf("RW version: %1.*s\n", |
||||
(int)sizeof(p->version_string_rw), |
||||
p->version_string_rw); |
||||
printf("Firmware copy: %s\n", |
||||
(p->current_image < |
||||
ARRAY_SIZE(ec_current_image_name) ? |
||||
ec_current_image_name[p->current_image] : |
||||
"?")); |
||||
ret = cros_ec_read_build_info(dev, &build_string); |
||||
if (!ret) |
||||
printf("Build info: %s\n", build_string); |
||||
} |
||||
} else if (0 == strcmp("ldo", cmd)) { |
||||
uint8_t index, state; |
||||
char *endp; |
||||
|
||||
if (argc < 3) |
||||
return CMD_RET_USAGE; |
||||
index = simple_strtoul(argv[2], &endp, 10); |
||||
if (*argv[2] == 0 || *endp != 0) |
||||
return CMD_RET_USAGE; |
||||
if (argc > 3) { |
||||
state = simple_strtoul(argv[3], &endp, 10); |
||||
if (*argv[3] == 0 || *endp != 0) |
||||
return CMD_RET_USAGE; |
||||
ret = cros_ec_set_ldo(udev, index, state); |
||||
} else { |
||||
ret = cros_ec_get_ldo(udev, index, &state); |
||||
if (!ret) { |
||||
printf("LDO%d: %s\n", index, |
||||
state == EC_LDO_STATE_ON ? |
||||
"on" : "off"); |
||||
} |
||||
} |
||||
|
||||
if (ret) { |
||||
debug("%s: Could not access LDO%d\n", __func__, index); |
||||
return ret; |
||||
} |
||||
} else { |
||||
return CMD_RET_USAGE; |
||||
} |
||||
|
||||
if (ret < 0) { |
||||
printf("Error: CROS-EC command failed (error %d)\n", ret); |
||||
ret = 1; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
U_BOOT_CMD( |
||||
crosec, 6, 1, do_cros_ec, |
||||
"CROS-EC utility command", |
||||
"init Re-init CROS-EC (done on startup automatically)\n" |
||||
"crosec id Read CROS-EC ID\n" |
||||
"crosec info Read CROS-EC info\n" |
||||
"crosec curimage Read CROS-EC current image\n" |
||||
"crosec hash Read CROS-EC hash\n" |
||||
"crosec reboot [rw | ro | cold] Reboot CROS-EC\n" |
||||
"crosec events Read CROS-EC host events\n" |
||||
"crosec clrevents [mask] Clear CROS-EC host events\n" |
||||
"crosec regioninfo <ro|rw> Read image info\n" |
||||
"crosec flashinfo Read flash info\n" |
||||
"crosec erase <ro|rw> Erase EC image\n" |
||||
"crosec read <ro|rw> <addr> [<size>] Read EC image\n" |
||||
"crosec write <ro|rw> <addr> [<size>] Write EC image\n" |
||||
"crosec vbnvcontext [hexstring] Read [write] VbNvContext from EC\n" |
||||
"crosec ldo <idx> [<state>] Switch/Read LDO state\n" |
||||
"crosec test run tests on cros_ec\n" |
||||
"crosec version Read CROS-EC version" |
||||
); |
@ -0,0 +1,86 @@ |
||||
/*
|
||||
* (C) Copyright 2016 Texas Instruments Incorporated, <www.ti.com> |
||||
* Keerthy <j-keerthy@ti.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <fdtdec.h> |
||||
#include <errno.h> |
||||
#include <dm.h> |
||||
#include <i2c.h> |
||||
#include <power/pmic.h> |
||||
#include <power/regulator.h> |
||||
#include <power/lp873x.h> |
||||
#include <dm/device.h> |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
static const struct pmic_child_info pmic_children_info[] = { |
||||
{ .prefix = "ldo", .driver = LP873X_LDO_DRIVER }, |
||||
{ .prefix = "buck", .driver = LP873X_BUCK_DRIVER }, |
||||
{ }, |
||||
}; |
||||
|
||||
static int lp873x_write(struct udevice *dev, uint reg, const uint8_t *buff, |
||||
int len) |
||||
{ |
||||
if (dm_i2c_write(dev, reg, buff, len)) { |
||||
error("write error to device: %p register: %#x!", dev, reg); |
||||
return -EIO; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int lp873x_read(struct udevice *dev, uint reg, uint8_t *buff, int len) |
||||
{ |
||||
if (dm_i2c_read(dev, reg, buff, len)) { |
||||
error("read error from device: %p register: %#x!", dev, reg); |
||||
return -EIO; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int lp873x_bind(struct udevice *dev) |
||||
{ |
||||
int regulators_node; |
||||
const void *blob = gd->fdt_blob; |
||||
int children; |
||||
int node = dev->of_offset; |
||||
|
||||
regulators_node = fdt_subnode_offset(blob, node, "regulators"); |
||||
|
||||
if (regulators_node <= 0) { |
||||
printf("%s: %s reg subnode not found!", __func__, dev->name); |
||||
return -ENXIO; |
||||
} |
||||
|
||||
children = pmic_bind_children(dev, regulators_node, pmic_children_info); |
||||
if (!children) |
||||
printf("%s: %s - no child found\n", __func__, dev->name); |
||||
|
||||
/* Always return success for this device */ |
||||
return 0; |
||||
} |
||||
|
||||
static struct dm_pmic_ops lp873x_ops = { |
||||
.read = lp873x_read, |
||||
.write = lp873x_write, |
||||
}; |
||||
|
||||
static const struct udevice_id lp873x_ids[] = { |
||||
{ .compatible = "ti,lp8732", .data = LP8732 }, |
||||
{ .compatible = "ti,lp8733" , .data = LP8733 }, |
||||
{ } |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(pmic_lp873x) = { |
||||
.name = "lp873x_pmic", |
||||
.id = UCLASS_PMIC, |
||||
.of_match = lp873x_ids, |
||||
.bind = lp873x_bind, |
||||
.ops = &lp873x_ops, |
||||
}; |
@ -0,0 +1,104 @@ |
||||
/*
|
||||
* (C) Copyright 2016 Texas Instruments Incorporated, <www.ti.com> |
||||
* Keerthy <j-keerthy@ti.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <fdtdec.h> |
||||
#include <errno.h> |
||||
#include <dm.h> |
||||
#include <i2c.h> |
||||
#include <power/pmic.h> |
||||
#include <power/regulator.h> |
||||
#include <power/palmas.h> |
||||
#include <dm/device.h> |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
static const struct pmic_child_info pmic_children_info[] = { |
||||
{ .prefix = "ldo", .driver = PALMAS_LDO_DRIVER }, |
||||
{ .prefix = "smps", .driver = PALMAS_SMPS_DRIVER }, |
||||
{ }, |
||||
}; |
||||
|
||||
static int palmas_write(struct udevice *dev, uint reg, const uint8_t *buff, |
||||
int len) |
||||
{ |
||||
if (dm_i2c_write(dev, reg, buff, len)) { |
||||
error("write error to device: %p register: %#x!", dev, reg); |
||||
return -EIO; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int palmas_read(struct udevice *dev, uint reg, uint8_t *buff, int len) |
||||
{ |
||||
if (dm_i2c_read(dev, reg, buff, len)) { |
||||
error("read error from device: %p register: %#x!", dev, reg); |
||||
return -EIO; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int palmas_bind(struct udevice *dev) |
||||
{ |
||||
int pmic_node = -1, regulators_node; |
||||
const void *blob = gd->fdt_blob; |
||||
int children; |
||||
int node = dev->of_offset; |
||||
int subnode, len; |
||||
|
||||
fdt_for_each_subnode(blob, subnode, node) { |
||||
const char *name; |
||||
char *temp; |
||||
|
||||
name = fdt_get_name(blob, subnode, &len); |
||||
temp = strstr(name, "pmic"); |
||||
if (temp) { |
||||
pmic_node = subnode; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
if (pmic_node <= 0) { |
||||
debug("%s: %s pmic subnode not found!", __func__, dev->name); |
||||
return -ENXIO; |
||||
} |
||||
|
||||
regulators_node = fdt_subnode_offset(blob, pmic_node, "regulators"); |
||||
|
||||
if (regulators_node <= 0) { |
||||
debug("%s: %s reg subnode not found!", __func__, dev->name); |
||||
return -ENXIO; |
||||
} |
||||
|
||||
children = pmic_bind_children(dev, regulators_node, pmic_children_info); |
||||
if (!children) |
||||
debug("%s: %s - no child found\n", __func__, dev->name); |
||||
|
||||
/* Always return success for this device */ |
||||
return 0; |
||||
} |
||||
|
||||
static struct dm_pmic_ops palmas_ops = { |
||||
.read = palmas_read, |
||||
.write = palmas_write, |
||||
}; |
||||
|
||||
static const struct udevice_id palmas_ids[] = { |
||||
{ .compatible = "ti,tps659038", .data = TPS659038 }, |
||||
{ .compatible = "ti,tps65917" , .data = TPS65917 }, |
||||
{ } |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(pmic_palmas) = { |
||||
.name = "palmas_pmic", |
||||
.id = UCLASS_PMIC, |
||||
.of_match = palmas_ids, |
||||
.bind = palmas_bind, |
||||
.ops = &palmas_ops, |
||||
}; |
@ -0,0 +1,137 @@ |
||||
/*
|
||||
* (C) Copyright 2016 Texas Instruments Incorporated, <www.ti.com> |
||||
* Keerthy <j-keerthy@ti.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <fdtdec.h> |
||||
#include <errno.h> |
||||
#include <dm.h> |
||||
#include <i2c.h> |
||||
#include <asm/gpio.h> |
||||
#include <power/pmic.h> |
||||
#include <power/regulator.h> |
||||
|
||||
#define GPIO_REGULATOR_MAX_STATES 2 |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
struct gpio_regulator_platdata { |
||||
struct gpio_desc gpio; /* GPIO for regulator voltage control */ |
||||
int states[GPIO_REGULATOR_MAX_STATES]; |
||||
int voltages[GPIO_REGULATOR_MAX_STATES]; |
||||
}; |
||||
|
||||
static int gpio_regulator_ofdata_to_platdata(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
struct gpio_regulator_platdata *dev_pdata; |
||||
struct gpio_desc *gpio; |
||||
const void *blob = gd->fdt_blob; |
||||
int node = dev->of_offset; |
||||
int ret, count, i, j; |
||||
u32 states_array[8]; |
||||
|
||||
dev_pdata = dev_get_platdata(dev); |
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
if (!uc_pdata) |
||||
return -ENXIO; |
||||
|
||||
/* Set type to gpio */ |
||||
uc_pdata->type = REGULATOR_TYPE_GPIO; |
||||
|
||||
/*
|
||||
* Get gpio regulator gpio desc |
||||
* Assuming one GPIO per regulator. |
||||
* Can be extended later to multiple GPIOs |
||||
* per gpio-regulator. As of now no instance with multiple |
||||
* gpios is presnt |
||||
*/ |
||||
gpio = &dev_pdata->gpio; |
||||
ret = gpio_request_by_name(dev, "gpios", 0, gpio, GPIOD_IS_OUT); |
||||
if (ret) |
||||
debug("regulator gpio - not found! Error: %d", ret); |
||||
|
||||
count = fdtdec_get_int_array_count(blob, node, "states", |
||||
states_array, 8); |
||||
|
||||
if (!count) |
||||
return -EINVAL; |
||||
|
||||
for (i = 0, j = 0; i < count; i += 2) { |
||||
dev_pdata->voltages[j] = states_array[i]; |
||||
dev_pdata->states[j] = states_array[i + 1]; |
||||
j++; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int gpio_regulator_get_value(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
struct gpio_regulator_platdata *dev_pdata = dev_get_platdata(dev); |
||||
int enable; |
||||
|
||||
if (!dev_pdata->gpio.dev) |
||||
return -ENOSYS; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
if (uc_pdata->min_uV > uc_pdata->max_uV) { |
||||
debug("Invalid constraints for: %s\n", uc_pdata->name); |
||||
return -EINVAL; |
||||
} |
||||
|
||||
enable = dm_gpio_get_value(&dev_pdata->gpio); |
||||
if (enable == dev_pdata->states[0]) |
||||
return dev_pdata->voltages[0]; |
||||
else |
||||
return dev_pdata->voltages[1]; |
||||
} |
||||
|
||||
static int gpio_regulator_set_value(struct udevice *dev, int uV) |
||||
{ |
||||
struct gpio_regulator_platdata *dev_pdata = dev_get_platdata(dev); |
||||
int ret; |
||||
bool enable; |
||||
|
||||
if (!dev_pdata->gpio.dev) |
||||
return -ENOSYS; |
||||
|
||||
if (uV == dev_pdata->voltages[0]) |
||||
enable = dev_pdata->states[0]; |
||||
else if (uV == dev_pdata->voltages[1]) |
||||
enable = dev_pdata->states[1]; |
||||
else |
||||
return -EINVAL; |
||||
|
||||
ret = dm_gpio_set_value(&dev_pdata->gpio, enable); |
||||
if (ret) { |
||||
error("Can't set regulator : %s gpio to: %d\n", dev->name, |
||||
enable); |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static const struct dm_regulator_ops gpio_regulator_ops = { |
||||
.get_value = gpio_regulator_get_value, |
||||
.set_value = gpio_regulator_set_value, |
||||
}; |
||||
|
||||
static const struct udevice_id gpio_regulator_ids[] = { |
||||
{ .compatible = "regulator-gpio" }, |
||||
{ }, |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(gpio_regulator) = { |
||||
.name = "gpio regulator", |
||||
.id = UCLASS_REGULATOR, |
||||
.ops = &gpio_regulator_ops, |
||||
.of_match = gpio_regulator_ids, |
||||
.ofdata_to_platdata = gpio_regulator_ofdata_to_platdata, |
||||
.platdata_auto_alloc_size = sizeof(struct gpio_regulator_platdata), |
||||
}; |
@ -0,0 +1,357 @@ |
||||
/*
|
||||
* (C) Copyright 2016 |
||||
* Texas Instruments Incorporated, <www.ti.com> |
||||
* |
||||
* Keerthy <j-keerthy@ti.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <fdtdec.h> |
||||
#include <errno.h> |
||||
#include <dm.h> |
||||
#include <i2c.h> |
||||
#include <power/pmic.h> |
||||
#include <power/regulator.h> |
||||
#include <power/lp873x.h> |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
static const char lp873x_buck_ctrl[LP873X_BUCK_NUM] = {0x2, 0x4}; |
||||
static const char lp873x_buck_volt[LP873X_BUCK_NUM] = {0x6, 0x7}; |
||||
static const char lp873x_ldo_ctrl[LP873X_LDO_NUM] = {0x8, 0x9}; |
||||
static const char lp873x_ldo_volt[LP873X_LDO_NUM] = {0xA, 0xB}; |
||||
|
||||
static int lp873x_buck_enable(struct udevice *dev, int op, bool *enable) |
||||
{ |
||||
int ret; |
||||
unsigned int adr; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
adr = uc_pdata->ctrl_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= LP873X_BUCK_MODE_MASK; |
||||
|
||||
if (ret) |
||||
*enable = true; |
||||
else |
||||
*enable = false; |
||||
|
||||
return 0; |
||||
} else if (op == PMIC_OP_SET) { |
||||
if (*enable) |
||||
ret |= LP873X_BUCK_MODE_MASK; |
||||
else |
||||
ret &= ~(LP873X_BUCK_MODE_MASK); |
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
if (ret) |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int lp873x_buck_volt2hex(int uV) |
||||
{ |
||||
if (uV > LP873X_BUCK_VOLT_MAX) |
||||
return -EINVAL; |
||||
else if (uV > 1400000) |
||||
return (uV - 1420000) / 20000 + 0x9E; |
||||
else if (uV > 730000) |
||||
return (uV - 735000) / 5000 + 0x18; |
||||
else if (uV >= 700000) |
||||
return (uV - 700000) / 10000 + 0x1; |
||||
else |
||||
return -EINVAL; |
||||
} |
||||
|
||||
static int lp873x_buck_hex2volt(int hex) |
||||
{ |
||||
if (hex > LP873X_BUCK_VOLT_MAX_HEX) |
||||
return -EINVAL; |
||||
else if (hex > 0x9D) |
||||
return 1400000 + (hex - 0x9D) * 20000; |
||||
else if (hex > 0x17) |
||||
return 730000 + (hex - 0x17) * 5000; |
||||
else if (hex >= 0x14) |
||||
return 700000 + (hex - 0x14) * 10000; |
||||
else |
||||
return -EINVAL; |
||||
} |
||||
|
||||
static int lp873x_buck_val(struct udevice *dev, int op, int *uV) |
||||
{ |
||||
unsigned int hex, adr; |
||||
int ret; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
if (op == PMIC_OP_GET) |
||||
*uV = 0; |
||||
|
||||
adr = uc_pdata->volt_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= LP873X_BUCK_VOLT_MASK; |
||||
ret = lp873x_buck_hex2volt(ret); |
||||
if (ret < 0) |
||||
return ret; |
||||
*uV = ret; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
hex = lp873x_buck_volt2hex(*uV); |
||||
if (hex < 0) |
||||
return hex; |
||||
|
||||
ret &= 0x0; |
||||
ret |= hex; |
||||
|
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
static int lp873x_ldo_enable(struct udevice *dev, int op, bool *enable) |
||||
{ |
||||
int ret; |
||||
unsigned int adr; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
adr = uc_pdata->ctrl_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= LP873X_LDO_MODE_MASK; |
||||
|
||||
if (ret) |
||||
*enable = true; |
||||
else |
||||
*enable = false; |
||||
|
||||
return 0; |
||||
} else if (op == PMIC_OP_SET) { |
||||
if (*enable) |
||||
ret |= LP873X_LDO_MODE_MASK; |
||||
else |
||||
ret &= ~(LP873X_LDO_MODE_MASK); |
||||
|
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
if (ret) |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int lp873x_ldo_volt2hex(int uV) |
||||
{ |
||||
if (uV > LP873X_LDO_VOLT_MAX) |
||||
return -EINVAL; |
||||
|
||||
return (uV - 800000) / 100000; |
||||
} |
||||
|
||||
static int lp873x_ldo_hex2volt(int hex) |
||||
{ |
||||
if (hex > LP873X_LDO_VOLT_MAX_HEX) |
||||
return -EINVAL; |
||||
|
||||
if (!hex) |
||||
return 0; |
||||
|
||||
return (hex * 100000) + 800000; |
||||
} |
||||
|
||||
static int lp873x_ldo_val(struct udevice *dev, int op, int *uV) |
||||
{ |
||||
unsigned int hex, adr; |
||||
int ret; |
||||
|
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
if (op == PMIC_OP_GET) |
||||
*uV = 0; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
adr = uc_pdata->volt_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= LP873X_LDO_VOLT_MASK; |
||||
ret = lp873x_ldo_hex2volt(ret); |
||||
if (ret < 0) |
||||
return ret; |
||||
*uV = ret; |
||||
return 0; |
||||
} |
||||
|
||||
hex = lp873x_ldo_volt2hex(*uV); |
||||
if (hex < 0) |
||||
return hex; |
||||
|
||||
ret &= ~LP873X_LDO_VOLT_MASK; |
||||
ret |= hex; |
||||
if (*uV > 1650000) |
||||
ret |= 0x80; |
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
static int lp873x_ldo_probe(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
uc_pdata->type = REGULATOR_TYPE_LDO; |
||||
|
||||
int idx = dev->driver_data; |
||||
if (idx >= LP873X_LDO_NUM) { |
||||
printf("Wrong ID for regulator\n"); |
||||
return -1; |
||||
} |
||||
|
||||
uc_pdata->ctrl_reg = lp873x_ldo_ctrl[idx]; |
||||
uc_pdata->volt_reg = lp873x_ldo_volt[idx]; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int ldo_get_value(struct udevice *dev) |
||||
{ |
||||
int uV; |
||||
int ret; |
||||
|
||||
ret = lp873x_ldo_val(dev, PMIC_OP_GET, &uV); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return uV; |
||||
} |
||||
|
||||
static int ldo_set_value(struct udevice *dev, int uV) |
||||
{ |
||||
return lp873x_ldo_val(dev, PMIC_OP_SET, &uV); |
||||
} |
||||
|
||||
static bool ldo_get_enable(struct udevice *dev) |
||||
{ |
||||
bool enable = false; |
||||
int ret; |
||||
|
||||
ret = lp873x_ldo_enable(dev, PMIC_OP_GET, &enable); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return enable; |
||||
} |
||||
|
||||
static int ldo_set_enable(struct udevice *dev, bool enable) |
||||
{ |
||||
return lp873x_ldo_enable(dev, PMIC_OP_SET, &enable); |
||||
} |
||||
|
||||
static int lp873x_buck_probe(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
int idx; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
uc_pdata->type = REGULATOR_TYPE_BUCK; |
||||
|
||||
idx = dev->driver_data; |
||||
if (idx >= LP873X_BUCK_NUM) { |
||||
printf("Wrong ID for regulator\n"); |
||||
return -1; |
||||
} |
||||
|
||||
uc_pdata->ctrl_reg = lp873x_buck_ctrl[idx]; |
||||
uc_pdata->volt_reg = lp873x_buck_volt[idx]; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int buck_get_value(struct udevice *dev) |
||||
{ |
||||
int uV; |
||||
int ret; |
||||
|
||||
ret = lp873x_buck_val(dev, PMIC_OP_GET, &uV); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return uV; |
||||
} |
||||
|
||||
static int buck_set_value(struct udevice *dev, int uV) |
||||
{ |
||||
return lp873x_buck_val(dev, PMIC_OP_SET, &uV); |
||||
} |
||||
|
||||
static bool buck_get_enable(struct udevice *dev) |
||||
{ |
||||
bool enable = false; |
||||
int ret; |
||||
|
||||
|
||||
ret = lp873x_buck_enable(dev, PMIC_OP_GET, &enable); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return enable; |
||||
} |
||||
|
||||
static int buck_set_enable(struct udevice *dev, bool enable) |
||||
{ |
||||
return lp873x_buck_enable(dev, PMIC_OP_SET, &enable); |
||||
} |
||||
|
||||
static const struct dm_regulator_ops lp873x_ldo_ops = { |
||||
.get_value = ldo_get_value, |
||||
.set_value = ldo_set_value, |
||||
.get_enable = ldo_get_enable, |
||||
.set_enable = ldo_set_enable, |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(lp873x_ldo) = { |
||||
.name = LP873X_LDO_DRIVER, |
||||
.id = UCLASS_REGULATOR, |
||||
.ops = &lp873x_ldo_ops, |
||||
.probe = lp873x_ldo_probe, |
||||
}; |
||||
|
||||
static const struct dm_regulator_ops lp873x_buck_ops = { |
||||
.get_value = buck_get_value, |
||||
.set_value = buck_set_value, |
||||
.get_enable = buck_get_enable, |
||||
.set_enable = buck_set_enable, |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(lp873x_buck) = { |
||||
.name = LP873X_BUCK_DRIVER, |
||||
.id = UCLASS_REGULATOR, |
||||
.ops = &lp873x_buck_ops, |
||||
.probe = lp873x_buck_probe, |
||||
}; |
@ -0,0 +1,453 @@ |
||||
/*
|
||||
* (C) Copyright 2016 |
||||
* Texas Instruments Incorporated, <www.ti.com> |
||||
* |
||||
* Keerthy <j-keerthy@ti.com> |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <fdtdec.h> |
||||
#include <errno.h> |
||||
#include <dm.h> |
||||
#include <i2c.h> |
||||
#include <power/pmic.h> |
||||
#include <power/regulator.h> |
||||
#include <power/palmas.h> |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
#define REGULATOR_ON 0x1 |
||||
#define REGULATOR_OFF 0x0 |
||||
|
||||
#define SMPS_MODE_MASK 0x3 |
||||
#define SMPS_MODE_SHIFT 0x0 |
||||
#define LDO_MODE_MASK 0x1 |
||||
#define LDO_MODE_SHIFT 0x0 |
||||
|
||||
static const char palmas_smps_ctrl[][PALMAS_SMPS_NUM] = { |
||||
{0x20, 0x24, 0x28, 0x2c, 0x30, 0x34, 0x38, 0x3c}, |
||||
{0x20, 0x24, 0x28, 0x2c, 0x30, 0x34, 0x38}, |
||||
{0x20, 0x24, 0x2c, 0x30, 0x38}, |
||||
}; |
||||
|
||||
static const char palmas_smps_volt[][PALMAS_SMPS_NUM] = { |
||||
{0x23, 0x27, 0x2b, 0x2f, 0x33, 0x37, 0x3b, 0x3c}, |
||||
{0x23, 0x27, 0x2b, 0x2f, 0x33, 0x37, 0x3b}, |
||||
{0x23, 0x27, 0x2f, 0x33, 0x3B} |
||||
}; |
||||
|
||||
static const char palmas_ldo_ctrl[][PALMAS_LDO_NUM] = { |
||||
{0x50, 0x52, 0x54, 0x56, 0x58, 0x5a, 0x5c, 0x5e, 0x60, 0x62, 0x64}, |
||||
{0x50, 0x52, 0x54, 0x56, 0x58, 0x5a, 0x5c, 0x5e, 0x60, 0x62, 0x64}, |
||||
{0x50, 0x52, 0x54, 0x5e, 0x62} |
||||
}; |
||||
|
||||
static const char palmas_ldo_volt[][PALMAS_LDO_NUM] = { |
||||
{0x51, 0x53, 0x55, 0x57, 0x59, 0x5b, 0x5d, 0x5f, 0x61, 0x63, 0x65}, |
||||
{0x51, 0x53, 0x55, 0x57, 0x59, 0x5b, 0x5d, 0x5f, 0x61, 0x63, 0x65}, |
||||
{0x51, 0x53, 0x55, 0x5f, 0x63} |
||||
}; |
||||
|
||||
static int palmas_smps_enable(struct udevice *dev, int op, bool *enable) |
||||
{ |
||||
int ret; |
||||
unsigned int adr; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
adr = uc_pdata->ctrl_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= PALMAS_SMPS_STATUS_MASK; |
||||
|
||||
if (ret) |
||||
*enable = true; |
||||
else |
||||
*enable = false; |
||||
|
||||
return 0; |
||||
} else if (op == PMIC_OP_SET) { |
||||
if (*enable) |
||||
ret |= PALMAS_SMPS_MODE_MASK; |
||||
else |
||||
ret &= ~(PALMAS_SMPS_MODE_MASK); |
||||
|
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
if (ret) |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int palmas_smps_volt2hex(int uV) |
||||
{ |
||||
if (uV > PALMAS_LDO_VOLT_MAX) |
||||
return -EINVAL; |
||||
|
||||
if (uV > 1650000) |
||||
return (uV - 1000000) / 20000 + 0x6; |
||||
|
||||
if (uV == 500000) |
||||
return 0x6; |
||||
else |
||||
return 0x6 + ((uV - 500000) / 10000); |
||||
} |
||||
|
||||
static int palmas_smps_hex2volt(int hex, bool range) |
||||
{ |
||||
unsigned int uV = 0; |
||||
|
||||
if (hex > PALMAS_SMPS_VOLT_MAX_HEX) |
||||
return -EINVAL; |
||||
|
||||
if (hex < 0x7) |
||||
uV = 500000; |
||||
else |
||||
uV = 500000 + (hex - 0x6) * 10000; |
||||
|
||||
if (range) |
||||
uV *= 2; |
||||
|
||||
return uV; |
||||
} |
||||
|
||||
static int palmas_smps_val(struct udevice *dev, int op, int *uV) |
||||
{ |
||||
unsigned int hex, adr; |
||||
int ret; |
||||
bool range; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
if (op == PMIC_OP_GET) |
||||
*uV = 0; |
||||
|
||||
adr = uc_pdata->volt_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
if (ret & PALMAS_SMPS_RANGE_MASK) |
||||
range = true; |
||||
else |
||||
range = false; |
||||
|
||||
ret &= PALMAS_SMPS_VOLT_MASK; |
||||
ret = palmas_smps_hex2volt(ret, range); |
||||
if (ret < 0) |
||||
return ret; |
||||
*uV = ret; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
hex = palmas_smps_volt2hex(*uV); |
||||
if (hex < 0) |
||||
return hex; |
||||
|
||||
ret &= ~PALMAS_SMPS_VOLT_MASK; |
||||
ret |= hex; |
||||
if (*uV > 1650000) |
||||
ret |= PALMAS_SMPS_RANGE_MASK; |
||||
|
||||
return pmic_reg_write(dev->parent, adr, ret); |
||||
} |
||||
|
||||
static int palmas_ldo_enable(struct udevice *dev, int op, bool *enable) |
||||
{ |
||||
int ret; |
||||
unsigned int adr; |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
adr = uc_pdata->ctrl_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= PALMAS_LDO_STATUS_MASK; |
||||
|
||||
if (ret) |
||||
*enable = true; |
||||
else |
||||
*enable = false; |
||||
|
||||
return 0; |
||||
} else if (op == PMIC_OP_SET) { |
||||
if (*enable) |
||||
ret |= PALMAS_LDO_MODE_MASK; |
||||
else |
||||
ret &= ~(PALMAS_LDO_MODE_MASK); |
||||
|
||||
ret = pmic_reg_write(dev->parent, adr, ret); |
||||
if (ret) |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int palmas_ldo_volt2hex(int uV) |
||||
{ |
||||
if (uV > PALMAS_LDO_VOLT_MAX) |
||||
return -EINVAL; |
||||
|
||||
return (uV - 850000) / 50000; |
||||
} |
||||
|
||||
static int palmas_ldo_hex2volt(int hex) |
||||
{ |
||||
if (hex > PALMAS_LDO_VOLT_MAX_HEX) |
||||
return -EINVAL; |
||||
|
||||
if (!hex) |
||||
return 0; |
||||
|
||||
return (hex * 50000) + 850000; |
||||
} |
||||
|
||||
static int palmas_ldo_val(struct udevice *dev, int op, int *uV) |
||||
{ |
||||
unsigned int hex, adr; |
||||
int ret; |
||||
|
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
|
||||
if (op == PMIC_OP_GET) |
||||
*uV = 0; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
adr = uc_pdata->volt_reg; |
||||
|
||||
ret = pmic_reg_read(dev->parent, adr); |
||||
if (ret < 0) |
||||
return ret; |
||||
|
||||
if (op == PMIC_OP_GET) { |
||||
ret &= PALMAS_LDO_VOLT_MASK; |
||||
ret = palmas_ldo_hex2volt(ret); |
||||
if (ret < 0) |
||||
return ret; |
||||
*uV = ret; |
||||
return 0; |
||||
} |
||||
|
||||
hex = palmas_ldo_volt2hex(*uV); |
||||
if (hex < 0) |
||||
return hex; |
||||
|
||||
ret &= ~PALMAS_LDO_VOLT_MASK; |
||||
ret |= hex; |
||||
if (*uV > 1650000) |
||||
ret |= 0x80; |
||||
|
||||
return pmic_reg_write(dev->parent, adr, ret); |
||||
} |
||||
|
||||
static int palmas_ldo_probe(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
struct udevice *parent; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
parent = dev_get_parent(dev); |
||||
int type = dev_get_driver_data(parent); |
||||
|
||||
uc_pdata->type = REGULATOR_TYPE_LDO; |
||||
|
||||
if (dev->driver_data) { |
||||
u8 idx = dev->driver_data - 1; |
||||
uc_pdata->ctrl_reg = palmas_ldo_ctrl[type][idx]; |
||||
uc_pdata->volt_reg = palmas_ldo_volt[type][idx]; |
||||
} else { |
||||
/* check for ldoln and ldousb cases */ |
||||
if (!strcmp("ldoln", dev->name)) { |
||||
uc_pdata->ctrl_reg = palmas_ldo_ctrl[type][9]; |
||||
uc_pdata->volt_reg = palmas_ldo_volt[type][9]; |
||||
} else if (!strcmp("ldousb", dev->name)) { |
||||
uc_pdata->ctrl_reg = palmas_ldo_ctrl[type][10]; |
||||
uc_pdata->volt_reg = palmas_ldo_volt[type][10]; |
||||
} |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int ldo_get_value(struct udevice *dev) |
||||
{ |
||||
int uV; |
||||
int ret; |
||||
|
||||
ret = palmas_ldo_val(dev, PMIC_OP_GET, &uV); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return uV; |
||||
} |
||||
|
||||
static int ldo_set_value(struct udevice *dev, int uV) |
||||
{ |
||||
return palmas_ldo_val(dev, PMIC_OP_SET, &uV); |
||||
} |
||||
|
||||
static bool ldo_get_enable(struct udevice *dev) |
||||
{ |
||||
bool enable = false; |
||||
int ret; |
||||
|
||||
ret = palmas_ldo_enable(dev, PMIC_OP_GET, &enable); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return enable; |
||||
} |
||||
|
||||
static int ldo_set_enable(struct udevice *dev, bool enable) |
||||
{ |
||||
return palmas_ldo_enable(dev, PMIC_OP_SET, &enable); |
||||
} |
||||
|
||||
static int palmas_smps_probe(struct udevice *dev) |
||||
{ |
||||
struct dm_regulator_uclass_platdata *uc_pdata; |
||||
struct udevice *parent; |
||||
int idx; |
||||
|
||||
uc_pdata = dev_get_uclass_platdata(dev); |
||||
|
||||
parent = dev_get_parent(dev); |
||||
int type = dev_get_driver_data(parent); |
||||
|
||||
uc_pdata->type = REGULATOR_TYPE_BUCK; |
||||
|
||||
switch (type) { |
||||
case PALMAS: |
||||
case TPS659038: |
||||
switch (dev->driver_data) { |
||||
case 123: |
||||
case 12: |
||||
uc_pdata->ctrl_reg = palmas_smps_ctrl[type][0]; |
||||
uc_pdata->volt_reg = palmas_smps_volt[type][0]; |
||||
break; |
||||
case 3: |
||||
uc_pdata->ctrl_reg = palmas_smps_ctrl[type][1]; |
||||
uc_pdata->volt_reg = palmas_smps_volt[type][1]; |
||||
break; |
||||
case 45: |
||||
uc_pdata->ctrl_reg = palmas_smps_ctrl[type][2]; |
||||
uc_pdata->volt_reg = palmas_smps_volt[type][2]; |
||||
break; |
||||
case 6: |
||||
case 7: |
||||
case 8: |
||||
case 9: |
||||
case 10: |
||||
idx = dev->driver_data - 4; |
||||
uc_pdata->ctrl_reg = palmas_smps_ctrl[type][idx]; |
||||
uc_pdata->volt_reg = palmas_smps_volt[type][idx]; |
||||
break; |
||||
|
||||
default: |
||||
printf("Wrong ID for regulator\n"); |
||||
} |
||||
break; |
||||
|
||||
case TPS65917: |
||||
switch (dev->driver_data) { |
||||
case 1: |
||||
case 2: |
||||
case 3: |
||||
case 4: |
||||
case 5: |
||||
idx = dev->driver_data - 1; |
||||
uc_pdata->ctrl_reg = palmas_smps_ctrl[type][idx]; |
||||
uc_pdata->volt_reg = palmas_smps_volt[type][idx]; |
||||
break; |
||||
|
||||
default: |
||||
printf("Wrong ID for regulator\n"); |
||||
} |
||||
break; |
||||
|
||||
default: |
||||
printf("Invalid PMIC ID\n"); |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int smps_get_value(struct udevice *dev) |
||||
{ |
||||
int uV; |
||||
int ret; |
||||
|
||||
ret = palmas_smps_val(dev, PMIC_OP_GET, &uV); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return uV; |
||||
} |
||||
|
||||
static int smps_set_value(struct udevice *dev, int uV) |
||||
{ |
||||
return palmas_smps_val(dev, PMIC_OP_SET, &uV); |
||||
} |
||||
|
||||
static bool smps_get_enable(struct udevice *dev) |
||||
{ |
||||
bool enable = false; |
||||
int ret; |
||||
|
||||
ret = palmas_smps_enable(dev, PMIC_OP_GET, &enable); |
||||
if (ret) |
||||
return ret; |
||||
|
||||
return enable; |
||||
} |
||||
|
||||
static int smps_set_enable(struct udevice *dev, bool enable) |
||||
{ |
||||
return palmas_smps_enable(dev, PMIC_OP_SET, &enable); |
||||
} |
||||
|
||||
static const struct dm_regulator_ops palmas_ldo_ops = { |
||||
.get_value = ldo_get_value, |
||||
.set_value = ldo_set_value, |
||||
.get_enable = ldo_get_enable, |
||||
.set_enable = ldo_set_enable, |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(palmas_ldo) = { |
||||
.name = PALMAS_LDO_DRIVER, |
||||
.id = UCLASS_REGULATOR, |
||||
.ops = &palmas_ldo_ops, |
||||
.probe = palmas_ldo_probe, |
||||
}; |
||||
|
||||
static const struct dm_regulator_ops palmas_smps_ops = { |
||||
.get_value = smps_get_value, |
||||
.set_value = smps_set_value, |
||||
.get_enable = smps_get_enable, |
||||
.set_enable = smps_set_enable, |
||||
}; |
||||
|
||||
U_BOOT_DRIVER(palmas_smps) = { |
||||
.name = PALMAS_SMPS_DRIVER, |
||||
.id = UCLASS_REGULATOR, |
||||
.ops = &palmas_smps_ops, |
||||
.probe = palmas_smps_probe, |
||||
}; |
@ -0,0 +1,19 @@ |
||||
#define LP8732 0x0 |
||||
#define LP8733 0x1 |
||||
|
||||
#define LP873X_LDO_NUM 2 |
||||
#define LP873X_BUCK_NUM 2 |
||||
|
||||
/* Drivers name */ |
||||
#define LP873X_LDO_DRIVER "lp873x_ldo" |
||||
#define LP873X_BUCK_DRIVER "lp873x_buck" |
||||
|
||||
#define LP873X_BUCK_VOLT_MASK 0xFF |
||||
#define LP873X_BUCK_VOLT_MAX_HEX 0xFF |
||||
#define LP873X_BUCK_VOLT_MAX 3360000 |
||||
#define LP873X_BUCK_MODE_MASK 0x1 |
||||
|
||||
#define LP873X_LDO_VOLT_MASK 0x1F |
||||
#define LP873X_LDO_VOLT_MAX_HEX 0x19 |
||||
#define LP873X_LDO_VOLT_MAX 3300000 |
||||
#define LP873X_LDO_MODE_MASK 0x1 |
@ -0,0 +1,25 @@ |
||||
#define PALMAS 0x0 |
||||
#define TPS659038 0x1 |
||||
#define TPS65917 0x2 |
||||
|
||||
/* I2C device address for pmic palmas */ |
||||
#define PALMAS_I2C_ADDR (0x12 >> 1) |
||||
#define PALMAS_LDO_NUM 11 |
||||
#define PALMAS_SMPS_NUM 8 |
||||
|
||||
/* Drivers name */ |
||||
#define PALMAS_LDO_DRIVER "palmas_ldo" |
||||
#define PALMAS_SMPS_DRIVER "palmas_smps" |
||||
|
||||
#define PALMAS_SMPS_VOLT_MASK 0x7F |
||||
#define PALMAS_SMPS_RANGE_MASK 0x80 |
||||
#define PALMAS_SMPS_VOLT_MAX_HEX 0x7F |
||||
#define PALMAS_SMPS_VOLT_MAX 3300000 |
||||
#define PALMAS_SMPS_MODE_MASK 0x3 |
||||
#define PALMAS_SMPS_STATUS_MASK 0x30 |
||||
|
||||
#define PALMAS_LDO_VOLT_MASK 0x3F |
||||
#define PALMAS_LDO_VOLT_MAX_HEX 0x3F |
||||
#define PALMAS_LDO_VOLT_MAX 3300000 |
||||
#define PALMAS_LDO_MODE_MASK 0x1 |
||||
#define PALMAS_LDO_STATUS_MASK 0x10 |
Loading…
Reference in new issue