uboot/drivers/misc/tegra186_bpmp.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Copyright (c) 2016, NVIDIA CORPORATION.
   4 */
   5
   6#include <common.h>
   7#include <dm.h>
   8#include <dm/lists.h>
   9#include <dm/root.h>
  10#include <mailbox.h>
  11#include <misc.h>
  12#include <asm/arch-tegra/bpmp_abi.h>
  13#include <asm/arch-tegra/ivc.h>
  14
  15#define BPMP_IVC_FRAME_COUNT 1
  16#define BPMP_IVC_FRAME_SIZE 128
  17
  18#define BPMP_FLAG_DO_ACK        BIT(0)
  19#define BPMP_FLAG_RING_DOORBELL BIT(1)
  20
  21DECLARE_GLOBAL_DATA_PTR;
  22
  23struct tegra186_bpmp {
  24        struct mbox_chan mbox;
  25        struct tegra_ivc ivc;
  26};
  27
  28static int tegra186_bpmp_call(struct udevice *dev, int mrq, void *tx_msg,
  29                              int tx_size, void *rx_msg, int rx_size)
  30{
  31        struct tegra186_bpmp *priv = dev_get_priv(dev);
  32        int ret, err;
  33        void *ivc_frame;
  34        struct mrq_request *req;
  35        struct mrq_response *resp;
  36        ulong start_time;
  37
  38        debug("%s(dev=%p, mrq=%u, tx_msg=%p, tx_size=%d, rx_msg=%p, rx_size=%d) (priv=%p)\n",
  39              __func__, dev, mrq, tx_msg, tx_size, rx_msg, rx_size, priv);
  40
  41        if ((tx_size > BPMP_IVC_FRAME_SIZE) || (rx_size > BPMP_IVC_FRAME_SIZE))
  42                return -EINVAL;
  43
  44        ret = tegra_ivc_write_get_next_frame(&priv->ivc, &ivc_frame);
  45        if (ret) {
  46                pr_err("tegra_ivc_write_get_next_frame() failed: %d\n", ret);
  47                return ret;
  48        }
  49
  50        req = ivc_frame;
  51        req->mrq = mrq;
  52        req->flags = BPMP_FLAG_DO_ACK | BPMP_FLAG_RING_DOORBELL;
  53        memcpy(req + 1, tx_msg, tx_size);
  54
  55        ret = tegra_ivc_write_advance(&priv->ivc);
  56        if (ret) {
  57                pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
  58                return ret;
  59        }
  60
  61        start_time = timer_get_us();
  62        for (;;) {
  63                ret = tegra_ivc_channel_notified(&priv->ivc);
  64                if (ret) {
  65                        pr_err("tegra_ivc_channel_notified() failed: %d\n", ret);
  66                        return ret;
  67                }
  68
  69                ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame);
  70                if (!ret)
  71                        break;
  72
  73                /* Timeout 20ms; roughly 10x current max observed duration */
  74                if ((timer_get_us() - start_time) > 20 * 1000) {
  75                        pr_err("tegra_ivc_read_get_next_frame() timed out (%d)\n",
  76                              ret);
  77                        return -ETIMEDOUT;
  78                }
  79        }
  80
  81        resp = ivc_frame;
  82        err = resp->err;
  83        if (!err && rx_msg && rx_size)
  84                memcpy(rx_msg, resp + 1, rx_size);
  85
  86        ret = tegra_ivc_read_advance(&priv->ivc);
  87        if (ret) {
  88                pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
  89                return ret;
  90        }
  91
  92        if (err) {
  93                pr_err("BPMP responded with error %d\n", err);
  94                /* err isn't a U-Boot error code, so don't that */
  95                return -EIO;
  96        }
  97
  98        return rx_size;
  99}
 100
 101/**
 102 * The BPMP exposes multiple different services. We create a sub-device for
 103 * each separate type of service, since each device must be of the appropriate
 104 * UCLASS.
 105 */
 106static int tegra186_bpmp_bind(struct udevice *dev)
 107{
 108        int ret;
 109        struct udevice *child;
 110
 111        debug("%s(dev=%p)\n", __func__, dev);
 112
 113        ret = device_bind_driver_to_node(dev, "tegra186_clk", "tegra186_clk",
 114                                         dev_ofnode(dev), &child);
 115        if (ret)
 116                return ret;
 117
 118        ret = device_bind_driver_to_node(dev, "tegra186_reset",
 119                                         "tegra186_reset", dev_ofnode(dev),
 120                                         &child);
 121        if (ret)
 122                return ret;
 123
 124        ret = device_bind_driver_to_node(dev, "tegra186_power_domain",
 125                                         "tegra186_power_domain",
 126                                         dev_ofnode(dev), &child);
 127        if (ret)
 128                return ret;
 129
 130        ret = dm_scan_fdt_dev(dev);
 131        if (ret)
 132                return ret;
 133
 134        return 0;
 135}
 136
 137static ulong tegra186_bpmp_get_shmem(struct udevice *dev, int index)
 138{
 139        int ret;
 140        struct fdtdec_phandle_args args;
 141        fdt_addr_t reg;
 142
 143        ret = fdtdec_parse_phandle_with_args(gd->fdt_blob, dev_of_offset(dev),
 144                                              "shmem", NULL, 0, index, &args);
 145        if (ret < 0) {
 146                pr_err("fdtdec_parse_phandle_with_args() failed: %d\n", ret);
 147                return ret;
 148        }
 149
 150        reg = fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, args.node,
 151                                                 "reg", 0, NULL, true);
 152        if (reg == FDT_ADDR_T_NONE) {
 153                pr_err("fdtdec_get_addr_size_auto_noparent() failed\n");
 154                return -ENODEV;
 155        }
 156
 157        return reg;
 158}
 159
 160static void tegra186_bpmp_ivc_notify(struct tegra_ivc *ivc)
 161{
 162        struct tegra186_bpmp *priv =
 163                container_of(ivc, struct tegra186_bpmp, ivc);
 164        int ret;
 165
 166        ret = mbox_send(&priv->mbox, NULL);
 167        if (ret)
 168                pr_err("mbox_send() failed: %d\n", ret);
 169}
 170
 171static int tegra186_bpmp_probe(struct udevice *dev)
 172{
 173        struct tegra186_bpmp *priv = dev_get_priv(dev);
 174        int ret;
 175        ulong tx_base, rx_base, start_time;
 176
 177        debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
 178
 179        ret = mbox_get_by_index(dev, 0, &priv->mbox);
 180        if (ret) {
 181                pr_err("mbox_get_by_index() failed: %d\n", ret);
 182                return ret;
 183        }
 184
 185        tx_base = tegra186_bpmp_get_shmem(dev, 0);
 186        if (IS_ERR_VALUE(tx_base)) {
 187                pr_err("tegra186_bpmp_get_shmem failed for tx_base\n");
 188                return tx_base;
 189        }
 190        rx_base = tegra186_bpmp_get_shmem(dev, 1);
 191        if (IS_ERR_VALUE(rx_base)) {
 192                pr_err("tegra186_bpmp_get_shmem failed for rx_base\n");
 193                return rx_base;
 194        }
 195        debug("shmem: rx=%lx, tx=%lx\n", rx_base, tx_base);
 196
 197        ret = tegra_ivc_init(&priv->ivc, rx_base, tx_base, BPMP_IVC_FRAME_COUNT,
 198                             BPMP_IVC_FRAME_SIZE, tegra186_bpmp_ivc_notify);
 199        if (ret) {
 200                pr_err("tegra_ivc_init() failed: %d\n", ret);
 201                return ret;
 202        }
 203
 204        tegra_ivc_channel_reset(&priv->ivc);
 205        start_time = timer_get_us();
 206        for (;;) {
 207                ret = tegra_ivc_channel_notified(&priv->ivc);
 208                if (!ret)
 209                        break;
 210
 211                /* Timeout 100ms */
 212                if ((timer_get_us() - start_time) > 100 * 1000) {
 213                        pr_err("Initial IVC reset timed out (%d)\n", ret);
 214                        ret = -ETIMEDOUT;
 215                        goto err_free_mbox;
 216                }
 217        }
 218
 219        return 0;
 220
 221err_free_mbox:
 222        mbox_free(&priv->mbox);
 223
 224        return ret;
 225}
 226
 227static int tegra186_bpmp_remove(struct udevice *dev)
 228{
 229        struct tegra186_bpmp *priv = dev_get_priv(dev);
 230
 231        debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
 232
 233        mbox_free(&priv->mbox);
 234
 235        return 0;
 236}
 237
 238static struct misc_ops tegra186_bpmp_ops = {
 239        .call = tegra186_bpmp_call,
 240};
 241
 242static const struct udevice_id tegra186_bpmp_ids[] = {
 243        { .compatible = "nvidia,tegra186-bpmp" },
 244        { }
 245};
 246
 247U_BOOT_DRIVER(tegra186_bpmp) = {
 248        .name           = "tegra186_bpmp",
 249        .id             = UCLASS_MISC,
 250        .of_match       = tegra186_bpmp_ids,
 251        .bind           = tegra186_bpmp_bind,
 252        .probe          = tegra186_bpmp_probe,
 253        .remove         = tegra186_bpmp_remove,
 254        .ops            = &tegra186_bpmp_ops,
 255        .priv_auto_alloc_size = sizeof(struct tegra186_bpmp),
 256};
 257