// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2016, NVIDIA CORPORATION. */ #include <common.h> #include <dm.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> #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), };