123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- #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) {
- error("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) {
- error("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) {
- error("tegra_ivc_channel_notified() failed: %d\n", ret);
- return ret;
- }
- ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame);
- if (!ret)
- break;
-
- if ((timer_get_us() - start_time) > 20 * 1000) {
- error("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) {
- error("tegra_ivc_write_advance() failed: %d\n", ret);
- return ret;
- }
- if (err) {
- error("BPMP responded with error %d\n", err);
-
- return -EIO;
- }
- return rx_size;
- }
- 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->of_offset, &child);
- if (ret)
- return ret;
- ret = device_bind_driver_to_node(dev, "tegra186_reset",
- "tegra186_reset", dev->of_offset,
- &child);
- if (ret)
- return ret;
- ret = device_bind_driver_to_node(dev, "tegra186_power_domain",
- "tegra186_power_domain",
- dev->of_offset, &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,
- "shmem", NULL, 0, index, &args);
- if (ret < 0) {
- error("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) {
- error("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)
- error("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) {
- error("mbox_get_by_index() failed: %d\n", ret);
- return ret;
- }
- tx_base = tegra186_bpmp_get_shmem(dev, 0);
- if (IS_ERR_VALUE(tx_base)) {
- error("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)) {
- error("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) {
- error("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;
-
- if ((timer_get_us() - start_time) > 100 * 1000) {
- error("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),
- };
|