| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Copyright (c) 2016, NVIDIA CORPORATION. |
| */ |
| |
| #include <common.h> |
| #include <dm.h> |
| #include <log.h> |
| #include <malloc.h> |
| #include <time.h> |
| #include <dm/lists.h> |
| #include <dm/root.h> |
| #include <mailbox.h> |
| #include <misc.h> |
| #include <asm/arch-tegra/bpmp_abi.h> |
| #include <asm/arch-tegra/ivc.h> |
| #include <linux/bitops.h> |
| #include <linux/err.h> |
| |
| #define BPMP_IVC_FRAME_COUNT 1 |
| #define BPMP_IVC_FRAME_SIZE 128 |
| |
| #define BPMP_FLAG_DO_ACK BIT(0) |
| #define BPMP_FLAG_RING_DOORBELL BIT(1) |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| struct tegra186_bpmp { |
| struct mbox_chan mbox; |
| struct tegra_ivc ivc; |
| }; |
| |
| static int tegra186_bpmp_call(struct udevice *dev, int mrq, void *tx_msg, |
| int tx_size, void *rx_msg, int rx_size) |
| { |
| struct tegra186_bpmp *priv = dev_get_priv(dev); |
| int ret, err; |
| void *ivc_frame; |
| struct mrq_request *req; |
| struct mrq_response *resp; |
| ulong start_time; |
| |
| debug("%s(dev=%p, mrq=%u, tx_msg=%p, tx_size=%d, rx_msg=%p, rx_size=%d) (priv=%p)\n", |
| __func__, dev, mrq, tx_msg, tx_size, rx_msg, rx_size, priv); |
| |
| if ((tx_size > BPMP_IVC_FRAME_SIZE) || (rx_size > BPMP_IVC_FRAME_SIZE)) |
| return -EINVAL; |
| |
| ret = tegra_ivc_write_get_next_frame(&priv->ivc, &ivc_frame); |
| if (ret) { |
| pr_err("tegra_ivc_write_get_next_frame() failed: %d\n", ret); |
| return ret; |
| } |
| |
| req = ivc_frame; |
| req->mrq = mrq; |
| req->flags = BPMP_FLAG_DO_ACK | BPMP_FLAG_RING_DOORBELL; |
| memcpy(req + 1, tx_msg, tx_size); |
| |
| ret = tegra_ivc_write_advance(&priv->ivc); |
| if (ret) { |
| pr_err("tegra_ivc_write_advance() failed: %d\n", ret); |
| return ret; |
| } |
| |
| start_time = timer_get_us(); |
| for (;;) { |
| ret = tegra_ivc_channel_notified(&priv->ivc); |
| if (ret) { |
| pr_err("tegra_ivc_channel_notified() failed: %d\n", ret); |
| return ret; |
| } |
| |
| ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame); |
| if (!ret) |
| break; |
| |
| /* Timeout 20ms; roughly 10x current max observed duration */ |
| if ((timer_get_us() - start_time) > 20 * 1000) { |
| pr_err("tegra_ivc_read_get_next_frame() timed out (%d)\n", |
| ret); |
| return -ETIMEDOUT; |
| } |
| } |
| |
| resp = ivc_frame; |
| err = resp->err; |
| if (!err && rx_msg && rx_size) |
| memcpy(rx_msg, resp + 1, rx_size); |
| |
| ret = tegra_ivc_read_advance(&priv->ivc); |
| if (ret) { |
| pr_err("tegra_ivc_write_advance() failed: %d\n", ret); |
| return ret; |
| } |
| |
| if (err) { |
| pr_err("BPMP responded with error %d\n", err); |
| /* err isn't a U-Boot error code, so don't that */ |
| return -EIO; |
| } |
| |
| return rx_size; |
| } |
| |
| /** |
| * The BPMP exposes multiple different services. We create a sub-device for |
| * each separate type of service, since each device must be of the appropriate |
| * UCLASS. |
| */ |
| static int tegra186_bpmp_bind(struct udevice *dev) |
| { |
| int ret; |
| struct udevice *child; |
| |
| debug("%s(dev=%p)\n", __func__, dev); |
| |
| ret = device_bind_driver_to_node(dev, "tegra186_clk", "tegra186_clk", |
| dev_ofnode(dev), &child); |
| if (ret) |
| return ret; |
| |
| ret = device_bind_driver_to_node(dev, "tegra186_reset", |
| "tegra186_reset", dev_ofnode(dev), |
| &child); |
| if (ret) |
| return ret; |
| |
| ret = device_bind_driver_to_node(dev, "tegra186_power_domain", |
| "tegra186_power_domain", |
| dev_ofnode(dev), &child); |
| if (ret) |
| return ret; |
| |
| ret = dm_scan_fdt_dev(dev); |
| if (ret) |
| return ret; |
| |
| return 0; |
| } |
| |
| static ulong tegra186_bpmp_get_shmem(struct udevice *dev, int index) |
| { |
| int ret; |
| struct fdtdec_phandle_args args; |
| fdt_addr_t reg; |
| |
| ret = fdtdec_parse_phandle_with_args(gd->fdt_blob, dev_of_offset(dev), |
| "shmem", NULL, 0, index, &args); |
| if (ret < 0) { |
| pr_err("fdtdec_parse_phandle_with_args() failed: %d\n", ret); |
| return ret; |
| } |
| |
| reg = fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, args.node, |
| "reg", 0, NULL, true); |
| if (reg == FDT_ADDR_T_NONE) { |
| pr_err("fdtdec_get_addr_size_auto_noparent() failed\n"); |
| return -ENODEV; |
| } |
| |
| return reg; |
| } |
| |
| static void tegra186_bpmp_ivc_notify(struct tegra_ivc *ivc) |
| { |
| struct tegra186_bpmp *priv = |
| container_of(ivc, struct tegra186_bpmp, ivc); |
| int ret; |
| |
| ret = mbox_send(&priv->mbox, NULL); |
| if (ret) |
| pr_err("mbox_send() failed: %d\n", ret); |
| } |
| |
| static int tegra186_bpmp_probe(struct udevice *dev) |
| { |
| struct tegra186_bpmp *priv = dev_get_priv(dev); |
| int ret; |
| ulong tx_base, rx_base, start_time; |
| |
| debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv); |
| |
| ret = mbox_get_by_index(dev, 0, &priv->mbox); |
| if (ret) { |
| pr_err("mbox_get_by_index() failed: %d\n", ret); |
| return ret; |
| } |
| |
| tx_base = tegra186_bpmp_get_shmem(dev, 0); |
| if (IS_ERR_VALUE(tx_base)) { |
| pr_err("tegra186_bpmp_get_shmem failed for tx_base\n"); |
| return tx_base; |
| } |
| rx_base = tegra186_bpmp_get_shmem(dev, 1); |
| if (IS_ERR_VALUE(rx_base)) { |
| pr_err("tegra186_bpmp_get_shmem failed for rx_base\n"); |
| return rx_base; |
| } |
| debug("shmem: rx=%lx, tx=%lx\n", rx_base, tx_base); |
| |
| ret = tegra_ivc_init(&priv->ivc, rx_base, tx_base, BPMP_IVC_FRAME_COUNT, |
| BPMP_IVC_FRAME_SIZE, tegra186_bpmp_ivc_notify); |
| if (ret) { |
| pr_err("tegra_ivc_init() failed: %d\n", ret); |
| return ret; |
| } |
| |
| tegra_ivc_channel_reset(&priv->ivc); |
| start_time = timer_get_us(); |
| for (;;) { |
| ret = tegra_ivc_channel_notified(&priv->ivc); |
| if (!ret) |
| break; |
| |
| /* Timeout 100ms */ |
| if ((timer_get_us() - start_time) > 100 * 1000) { |
| pr_err("Initial IVC reset timed out (%d)\n", ret); |
| ret = -ETIMEDOUT; |
| goto err_free_mbox; |
| } |
| } |
| |
| return 0; |
| |
| err_free_mbox: |
| mbox_free(&priv->mbox); |
| |
| return ret; |
| } |
| |
| static int tegra186_bpmp_remove(struct udevice *dev) |
| { |
| struct tegra186_bpmp *priv = dev_get_priv(dev); |
| |
| debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv); |
| |
| mbox_free(&priv->mbox); |
| |
| return 0; |
| } |
| |
| static struct misc_ops tegra186_bpmp_ops = { |
| .call = tegra186_bpmp_call, |
| }; |
| |
| static const struct udevice_id tegra186_bpmp_ids[] = { |
| { .compatible = "nvidia,tegra186-bpmp" }, |
| { } |
| }; |
| |
| U_BOOT_DRIVER(tegra186_bpmp) = { |
| .name = "tegra186_bpmp", |
| .id = UCLASS_MISC, |
| .of_match = tegra186_bpmp_ids, |
| .bind = tegra186_bpmp_bind, |
| .probe = tegra186_bpmp_probe, |
| .remove = tegra186_bpmp_remove, |
| .ops = &tegra186_bpmp_ops, |
| .priv_auto_alloc_size = sizeof(struct tegra186_bpmp), |
| }; |